std:: is_constant_evaluated
|
|
|||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
|
ヘッダーで定義
<type_traits>
|
||
|
constexpr
bool
is_constant_evaluated
(
)
noexcept
;
|
(C++20以降) | |
関数呼び出しが定数評価コンテキスト内で発生するかどうかを検出します。呼び出しの評価が 明示的に定数評価される 式または変換の評価内で行われる場合は true を返します。それ以外の場合は false を返します。
以下の変数の初期化子が明示的に定数評価されるかどうかを判断するために、コンパイラは最初に試行的な定数評価を実行する場合があります:
- 参照型またはconst修飾された整数型もしくは列挙型の変数;
- static変数およびthread_local変数。
この場合、結果に依存することは推奨されません。
int y = 0; const int a = std::is_constant_evaluated() ? y : 1; // 定数評価の試行は失敗します。定数評価は破棄されます。 // 変数aは1で動的に初期化されます const int b = std::is_constant_evaluated() ? 2 : y; // std::is_constant_evaluated() == true での定数評価は成功します。 // 変数bは2で静的に初期化されます
目次 |
パラメータ
(なし)
戻り値
true 呼び出しの評価が、明示的に定数評価される式または変換の評価内で行われる場合;それ以外の場合 false 。
実装例
// この実装はC++23のif constevalを必要とします constexpr bool is_constant_evaluated() noexcept { if consteval { return true; } else { return false; } } |
注記
static_assert 宣言の条件または constexpr if文 の条件として直接使用された場合、 std :: is_constant_evaluated ( ) は常に true を返します。
C++20では
if consteval
が存在しないため、
std::is_constant_evaluated
は通常コンパイラ拡張を使用して実装されます。
| 機能テスト マクロ | 値 | 標準 | 機能 |
|---|---|---|---|
__cpp_lib_is_constant_evaluated
|
201811L
|
(C++20) |
std::is_constant_evaluated
|
例
#include <cmath> #include <iostream> #include <type_traits> constexpr double power(double b, int x) { if (std::is_constant_evaluated() && !(b == 0.0 && x < 0)) { // 定数評価コンテキスト: constexpr対応アルゴリズムを使用 if (x == 0) return 1.0; double r {1.0}; double p {x > 0 ? b : 1.0 / b}; for (auto u = unsigned(x > 0 ? x : -x); u != 0; u /= 2) { if (u & 1) r *= p; p *= p; } return r; } else { // コードジェネレータに処理を任せる return std::pow(b, double(x)); } } int main() { // 定数式コンテキスト constexpr double kilo = power(10.0, 3); int n = 3; // 定数式ではない(nは定数式コンテキストで右辺値に変換できないため) // std::pow(10.0, double(n)) と同等 double mucho = power(10.0, n); std::cout << kilo << " " << mucho << "\n"; // (3) }
出力:
1000 1000
関連項目
constexpr
指定子
(C++11)
|
変数または関数の値がコンパイル時に計算可能であることを指定する |
consteval
指定子
(C++20)
|
関数が 即時関数 であることを指定する。つまり、関数へのすべての呼び出しは定数評価で行われなければならない |
constinit
指定子
(C++20)
|
変数が静的初期化( ゼロ初期化 および 定数初期化 )を持つことを表明する |