const 和 constexpr 深入学习

在 C++ 中,const 和 constexpr 都可以用来修饰对象和函数。修饰对象时,const 表示它是常量,而 constexpr 表示它是一个常量表达式。常量表达式必须在编译时期被计算1。修饰函数时,const 只能用于非静态成员的函数,而 constexpr 可以用于含参和无参函数。constexpr 函数适用于常量表达式,只有在下面的情况下编译器才会接受 constexpr 函数:

  1. 函数体必须足够简单,除了 typedef 和静态元素,只允许有 return 语句。如构造函数只能有初始化列表,typedef 和静态元素(实际上在 C++14 标准中已经允许定义语句存在于 constexpr 函数体内了)。
  2. 参数和返回值必须是字面值类型。

常量表达式的概念如下:

  • 必须是可以在编译阶段被识别的。比如模版的参数/数组的大小。
  • constexpr 修饰某物并不保证它一定在编译时被计算,也可以在运行时被计算。
  • 不用 constexpr 也可能是一个常量表达式,如 const int N = 3; int num [N] = {1,2,3,};N 在声明时初始化,满足常量表达式的标准,虽然没用 constexpr

所以,到底什么时候用 constexpr?像上面的 N,虽然没有用 constexpr 仍然是一个常量表达式。事实上,const 本身和它的枚举类型若在声明时初始化那么就是一个常量表达式。在函数中若有常量表达式那么必须用 constexpr,仅仅满足常量表达式的条件是不够的。

所以,constconstexpr 的主要区别在于:

  • const 变量的初始化可以延迟到运行时,而 constexpr 变量必须在编译时进行初始化。
  • const 变量可以通过 const_cast 类型转换来修改值,而 constexpr 是不可以修改的。可以将 const 理解为只读变量更符合其含义。
  • const 只能用于非静态成员函数,而 constexpr 可以和成员、非成员、构造函数一起使用。

我们来深入学习代码


int fun(int x) {

	return x;
}

constexpr int ff(int x) {

	return x;
}

int main()
{
	
	int x = 10;
	const int a = x;
	constexpr int b = x;  // 错误的 表达式 必须是常量
	constexpr int c = fun(10);   // 也是错误的
	constexpr int d = ff(10);   // 正确的
	constexpr int d = ff(x);   // 错误的
}

 

  1. const int a = x; 是正确的,因为 a 是一个常量,可以被赋值为一个变量。

  2. constexpr int b = x; 是错误的,因为 x 不是常量表达式,而 constexpr 变量必须在编译时求值为常量表达式。

  3. constexpr int c = fun(10); 是错误的,因为 fun(10) 返回的不是常量表达式。在 constexpr 上下文中,函数的返回值必须是常量表达式。

  4. constexpr int d = ff(10); 是正确的,因为 ff(10) 返回的是常量表达式。

  5. constexpr int e = ff(x); 是错误的,因为 x 不是常量表达式。

  • 38
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

wniuniu_

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值