C++ const 和 constexpr 的区别

本文探讨了constexpr在C++中的作用,它如何区分编译期常量与运行期常量,并通过实例说明了如何检测函数是否产生编期常量。同时,讲述了constexpr函数的限制和其在模板特化中的应用价值。

对于修饰Object来说,

const并未区分出编译期常量和运行期常量

constexpr限定在了编译期常量

然后我想对修饰函数多说两句,那就是constexpr修饰的函数,返回值不一定是编译期常量。#It is not a bug, it is a feature.#

constexpr int foo(int i)
{
    return i + 5;
}
int main()
{
    int i = 10;
    std::array<int, foo(5)> arr; // OK

    foo(i); // Call is Ok

    // But...
    //std::array<int, foo(i)> arr1; // Error

    system("pause");


    return 0;

}

如果把constexpr 去掉,变成如下

int foo(int i)   
{
    return i + 5;
}
int main()
{
    int i = 10;
    std::array<int, foo(5)> arr; // 关于c++报错:“表达式必须含有常量值” 

    foo(i); // Call is Ok

    // But...
    //std::array<int, foo(i)> arr1; // Error

    system("pause");


    return 0;

}

所以,对于constexpr需要两方面看待。

constexpr修饰的函数,简单的来说,如果其传入的参数可以在编译时期计算出来,那么这个函数就会产生编译时期的值。但是,传入的参数如果不能在编译时期计算出来,那么constexpr修饰的函数就和普通函数一样了。不过,我们不必因此而写两个版本,所以如果函数体适用于constexpr函数的条件,可以尽量加上constexpr。

而检测constexpr函数是否产生编译时期值的方法很简单,就是利用std::array需要编译期常值才能编译通过的小技巧。这样的话,即可检测你所写的函数是否真的产生编译期常值了。

这样的函数有什么用呢?

比如你想特化一个模板,在传入的模板参数是std::numeric_limits<int>::max()时报错。那么在没有constexpr之前,就没救了,只能用INT_MAX宏。

const是变量类型名的一部分, part of type name,一个名字叫“const T”或者“T const”的类型,和T这个类型本身处于一种平级的关系,和T不同的就在于这个类型的对象自产生后就不能再更改了。

constexpr是声明的一部分,即 part of a declaration,他不是变量类型的一部分。当他出现在一个变量的声明中时,他要求编译器在编译期间就初始化并确定下来这个变量(否则编译报错);当他出现在一个函数的声明中时,他要求至少有一条return路径可以(但不是必须)在编译中确定下来,即返回的是编译期常量。

二者的联系就在于,在使用constexpr声明一个类型为T的变量时,constexpr会自动把这个变量定义为const T类型。即constexpr在完成它本职工作(告诉编译器这是个编译期常量)的同时,还把原来的T类型改为了const T类型。这就是二者的联系。



作者:小天狼星不来客
链接:https://www.zhihu.com/question/35614219/answer/1477686611
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。


作者:蓝色
链接:https://www.zhihu.com/question/35614219/answer/63798713
来源:知乎
著作权归作者所有。商业转载请联系作者获得授权,非商业转载请注明出处。

### C++ 中 `const` 与 `constexpr` 的区别使用场景 #### 定义与基本概念 `const` `constexpr` 都用于表示不可变的数据,但在语义用途上有显著差异。`const` 主要用来表达“只读”的含义,而 `constexpr` 则更进一步,强调数据可以在编译期计算并确定其值[^1]。 #### 编译期与运行期的支持 - **`const`**: 表示变量在初始化之后不能再被修改。它的值可能是在运行时才确定的,因此不一定能在编译期求值。 - **`constexpr`**: 不仅具备 `const` 的不可修改特性,还能确保该值在编译期就被计算出来。这意味着它适用于需要静态常量的场合,比如数组大小或模板参数等[^3]。 #### 类型支持 - **`const`**: 可以应用于任何类型的对象,无论是基础类型还是复杂类的对象。 - **`constexpr`**: 对于函数返回值或者变量初始值有严格的要求——它们必须能够通过简单的表达式来计算得出,并且这些操作数也应该是字面量(literals),或者是其他已经声明为 `constexpr` 的实体[^4]。 #### 使用建议 为了更好地利用这两种关键字的不同特点,在实际编码实践中可以遵循如下原则: - 当只需要表明某个东西不应该改变的时候(即使这个不变性只是逻辑上的),应该优先考虑使用 `const` 关键字; - 如果希望某项数值不仅保持恒定而且能够在编译期间得到解析,则应当选用 `constexpr`[^1]。 以下是两个简单例子展示如何正确运用这两个关键词: ```cpp // Example of using 'const' const int max_size = 10; // Runtime constant, cannot be used as non-type template parameter. // Example of using 'constexpr' constexpr double pi = 3.14159265358979323846; // Compile-time constant. ``` #### 总结 虽然两者都能创建不可更改的变量,但因各自特性的不同决定了各自的适用范围。对于那些仅仅需要保护免受意外改动影响的情况来说,采用 `const` 就足够了;而对于那些确实有必要提前知道确切数值的情形下,则更适合应用 `constexpr`[^2]。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值