C++那些事之const全家桶

C++那些事之const全家桶

大纲

  • const

  • constexpr

  • C++20

    • consteval

    • constinit

注:本文所有代码示例已放星球。

1.const

const 是我们在学习 C++/C 时最早接触的关键字之一,通常作为 cv-qualifier 使用,表示对象不可变。它并不保证 编译时评估,具体何时评估由编译器决定。例如:

const int a = 1; // 编译时
const auto x = getdata(); // 运行时

const有时可以用于编译时常量表达式,但是有一些例外,例如:

const double count_1 = 3.3;
std::array<double, static_cast<int>(count_1)> moreDoubles1 {1.1, 2.2, 3.3};

此时便会报错:the value of 'count_1' is not usable in a constant expression

但是这样是正常的:

const int count = 3;
std::array<double, count> doubles {1.1, 2.2, 3.3};

本质在于编译时表达式求值,于是C++11引入了constexpr。

2.constexpr

constexpr 可以用来指定变量的值可以出现在常量表达式,此时只需要将前面例子的const变为constexpr,便可以正常运行。

constexpr double count_2 = 3.3;
std::array<double, static_cast<int>(count_2)> moreDoubles2 {1.1, 2.2, 3.3};

constexpr 并不一定只能在编译时执行。它的本质含义是 可以 在编译时计算,但如果在运行时使用 constexpr 变量或函数,它也可以在运行时执行。

例如:result1与result2完全不一样,result1是编译时计算,但是result2是运行时计算的,抛个疑问:你知道为什么吗?

所以并不是指定了constexpr的函数它就一定是编译时计算的。

constexpr int compileTimeSum(int a, int b) {
    return a + b;
}

int main() {
    constexpr int result1 = compileTimeSum(3, 5); 
    int x = 10;
    int result2 = compileTimeSum(x, 20);  
    return 0;
}

那么谁能保证呢?于是C++20来了。

3.C++20

3.1 consteval

consteval解决了上面的问题,它只能用于函数,并强制所有调用在编译时进行。如果我们此时将上面的constexpr改为consteval,此时result2会发生什么呢?

那便是报错:call to consteval function 'compileTimeSum(x, 20)' is not a constant expression

consteval int compileTimeSum(int a, int b) {
    return a + b;
}

int main() {
    constexpr int result1 = compileTimeSum(3, 5); 
    int x = 10;
    int result2 = compileTimeSum(x, 20);  
    return 0;
}

如果使用了consteval使用在变量上,也会报错:error: a variable cannot be declared 'consteval'。

https://en.cppreference.com/w/cpp/language/consteval

3.2 constinit

constinit可以强制静态或线程局部变量进行常量初始化。

  • 不能与 constexpr、consteval 一起使用

  • 一个非类型说明符,作用是assert

  • 保证变量是静态初始化

  • 只能用于 static 或者 thread_local

例如:下面会编译报错:error: 'constinit' can only be applied to a variable with static or thread storage duration

constinit int a = 100;  
int main() {  
      ++a; 
      constinit auto b = 100;  // error
      // b has thread storage duration
      constinit thread_local auto b = 100;
}

一起探索更多C++项目/知识~

9005cb3e1a2bf0dbf3cd1525c7d84fa0.png

往期推荐:

向量数据库milvus源码剖析之开篇

热度更新,手把手实现工业级线程池

玩转cpp小项目星球3周年了!

cb7819e4073bcf1db7562b06b5de7b88.jpeg

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值