質問をすることでしか得られない、回答やアドバイスがある。

15分調べてもわからないことは、質問しよう!

新規登録して質問してみよう
ただいま回答率
85.48%
C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

Q&A

解決済

2回答

1882閲覧

c++20 constexpr関数の引数をconstexprかどうか判定するには?

bjnes

総合スコア113

C++

C++はC言語をもとにしてつくられた最もよく使われるマルチパラダイムプログラミング言語の1つです。オブジェクト指向、ジェネリック、命令型など広く対応しており、多目的に使用されています。

0グッド

0クリップ

投稿2020/08/11 09:08

編集2020/08/11 09:09

前提・実現したいこと

下記ソースコードのようにobjがconstexprになるような型にしたい。
C++20の仕様上この文法は可能なのでしょうか?

発生している問題・エラーメッセージ

‘obj’ is not a constant expression

該当のソースコード

C++

1 2 template<class T> 3 concepts IsConstexprType = ?????????????????? //ここがわからない 4 5   template <class T> 6 static constexpr auto FitSizeTypeSelector(T obj) 7 { 8 if constexpr(IsConstexprType<T>) 9 { 10 constexpr auto Uint8Max = std::numeric_limits<uint8_t>::max(); 11 constexpr auto Uint16Max = std::numeric_limits<uint16_t>::max(); 12 constexpr auto Uint32Max = std::numeric_limits<uint32_t>::max(); 13 constexpr auto Uint64Max = std::numeric_limits<uint64_t>::max(); 14 if constexpr (static_cast<uint8_t>(obj) <= Uint8Max) return (uint8_t{}); 15 if constexpr (static_cast<uint16_t>(obj) <= Uint16Max) return (uint16_t{}); 16 if constexpr (static_cast<uint32_t>(obj) <= Uint32Max) return (uint32_t{}); 17 if constexpr (static_cast<uint64_t>(obj) <= Uint64Max) return (uint64_t{}); 18 } 19 20 return uint{}; 21 22 } 23 24//使い方 25 using Mystrage = decltype(FitSizeTypeSelector(100)); 26

補足情報(FW/ツールのバージョンなど)

gcc -v
組み込み spec を使用しています。
COLLECT_GCC=gcc
COLLECT_LTO_WRAPPER=/usr/lib/gcc/x86_64-pc-linux-gnu/10.1.0/lto-wrapper
ターゲット: x86_64-pc-linux-gnu
configure 設定: /build/gcc/src/gcc/configure --prefix=/usr --libdir=/usr/lib --libexecdir=/usr/lib --mandir=/usr/share/man --infodir=/usr/share/info --with-bugurl=https://bugs.archlinux.org/ --enable-languages=c,c++,ada,fortran,go,lto,objc,obj-c++,d --with-isl --with-linker-hash-style=gnu --with-system-zlib --enable-__cxa_atexit --enable-cet=auto --enable-checking=release --enable-clocale=gnu --enable-default-pie --enable-default-ssp --enable-gnu-indirect-function --enable-gnu-unique-object --enable-install-libiberty --enable-linker-build-id --enable-lto --enable-multilib --enable-plugin --enable-shared --enable-threads=posix --disable-libssp --disable-libstdcxx-pch --disable-libunwind-exceptions --disable-werror gdc_include_dir=/usr/include/dlang/gdc
スレッドモデル: posix
Supported LTO compression algorithms: zlib zstd
gcc バージョン 10.1.0 (GCC)

気になる質問をクリップする

クリップした質問は、後からいつでもMYページで確認できます。

またクリップした質問に回答があった際、通知やメールを受け取ることができます。

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

guest

回答2

0

入力が定数式であるかどうかは式が定数式の性質を持つかどうかであって型の情報ではありません。 質問の事例においては T の型はあくまでも int だという情報しかないのです。 つまり T を検査して定数式かどうかを知ることは出来ません。

入力が定数式かどうかではなくコンパイル時に呼ばれたかどうかを判定する std::is_constant_evaluated() という関数が用意されているので、これを用いれば定数式かどうかで処理を切り替えることは出来ますが、質問の事例は結果の型も切り替えたいということのようなので簡単には実現できなさそうです。

投稿2020/08/11 13:13

SaitoAtsushi

総合スコア5444

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

bjnes

2020/08/14 01:13

ありがとうございます。 ダメ元で質問をしていますので、申し訳ありませんでした。
guest

0

ベストアンサー

(SaitoAtsushiさん回答とは異なる側面からの情報提供です)

C++20からコンパイル時評価を強制するconsteval関数が導入されます。もしかしたら求める動作に近いことを表現できるかもしれません。解説記事「さらに出来るようになったconstexpr(C++20)」もご参考に。


本題から外れますが、質問文中の関数実装では常にuint8_tを返します。式static_cast<uint8_t>(obj) <= Uint8Maxは常にtrueです。

C++

1constexpr auto Uint8Max = std::numeric_limits<uint8_t>::max(); 2if constexpr (static_cast<uint8_t>(obj) <= Uint8Max) return (uint8_t{});

投稿2020/08/13 09:03

編集2020/08/13 09:04
yohhoy

総合スコア6191

バッドをするには、ログインかつ

こちらの条件を満たす必要があります。

bjnes

2020/08/14 01:12

私の意図を裏読みしていただいて、ありがとうございます。 たしかに、この関数の使用される意図の延長上にはコンパイル時のみ 動作させたいという意図があります。 それは、コンパイル時なら最も小さいサイズにして、 実行時の一時変数(スタック変数)ならば、オーバーフロー防止の為、 大きいサイズにしたいという意図があります。 ご指摘ありがとうございます。
guest

あなたの回答

tips

太字

斜体

打ち消し線

見出し

引用テキストの挿入

コードの挿入

リンクの挿入

リストの挿入

番号リストの挿入

表の挿入

水平線の挿入

プレビュー

15分調べてもわからないことは
teratailで質問しよう!

ただいまの回答率
85.48%

質問をまとめることで
思考を整理して素早く解決

テンプレート機能で
簡単に質問をまとめる

質問する

関連した質問