运行时常量(const)和编译时常量(constexpr)

在c++中,我们会常常遇到常量的概念。常量表示该值不可修改。如:

const int i=3;//i为常量

const 还可以修饰函数参数,函数返回值,函数本身,类等。在不同的条件下,const有不同的意义,大多数情况const描述的都是”运行时常量概念“,既具有运行时数据不可更改性。不过有时候,我们需要编译时期的常量性,这是const关键字无法保证的。如:

const int getconst()
{
	return 1;
}
void testfun(int n)
{
	int arr[getconst()] = { 0 };//无法通过编译
	switch (n)
	{
	case getconst()://无法通过编译
	{
        //.......
	}break;
	default:
		break;
	}
}

那如果想要获得编译时期的常量,有没有办法呢?最简单的方法,可以使用c语言中的宏代替getconst函数。

#define getconst 1

当然这种简单粗暴的做法即使有效,也会把C++来回“石器时代”。C++11中对编译时期常量 引入了constexpr ,既常量表达式(const expression)。上面的getconst函数就可以用下面的声明方法:

constexpr int getconst()
{
	return 1;
}

即在函数表达式前面加上constexpr关键字即可。有了常量表达式这样的声明,编译器就可以在编译时期对getconst表达式进行值计算,从而将其视为一个编译时期的常量(可能)。在C++11中,常量表达式实际上不仅仅用于函数,还可以作用于数据声明,类的构造等。


1.constexpr修饰函数--常量表达式函数

在函数返回值类型前加上关键字constexpr来使其成为常量表达式函数。不过需要注意:

a.函数必须返回值(不能是void函数 ,也没意义)。  如:constexpr void f();这样的不返回值得函数就不能是常量表达式。因为无法获得常量得常量表达式时不被认可得。

b.在使用前必须已有定义。   对于普通函数而言。调用函数只需要有函数的声明就够了,但是常量表达式函数的 使用 不同。使用讲的时编译时值的计算,而调用时运行时期概念

constexpr int f();//声明
int a = f();//调用 运行时期
const int b = f();//调用
constexpr int c = f();//无法通过编译 编译器无法确定f()值     使用 编译时期
constexpr int f() { return 1; };//定义
constexpr int d = f();//使用

c.return 返回值语句表达式不能使用非常量表达式的函数,全局函数,且必须是一个常量表达式

const int e(){return 1;};
constexpr int f(){return e();};//无法通过编译

或形如:

int g =1;
constexpr int f(){return g;};//无法通过编译

2.constexpr修饰变量 -- 常量表达式值

a.修饰基本数据类型

通常情况下,常量表达时值必须被一个常量表达式赋值,即在使用前必须被初始化。

const int a=1;
constexpr int b =1;

多数情况下上面两者没有区别,但是a在全局中编译器会为a产生值,而b不一定会为其产生值,仅将其当做编译时期的值,只有代码显示地使用了它的值才会产生值。

b.修饰自定义类型 

当修饰自定义类型时注意两点:一是 构造函数加constexpr修饰 函数体必须为空;二是 初始化列表只能有常量表达式来赋值如:

	struct MyStruct
	{
	constexpr MyStruct(int x) :i(x) {};
        constexpr int getivalue(){return i};
        private:
            int i;
	};
	constexpr MyStruct ms(1);
	constexpr MyStruct ms2 = { 2 };

总结:constexpr体现的是编译时期的“常量性”, 如不能作用于virtual的成员函数

  • 3
    点赞
  • 14
    收藏
    觉得还不错? 一键收藏
  • 2
    评论
constconstexpr都是C++中用来定义常量的关键字。 const用于声明一个不可改变的常量。一旦被初始化后,它的值就不能再被修改。可以用来修饰变量、函数参数和函数返回值等。 constexpr也用于定义常量,但是它的特点是在编译期间就可以计算出它的值。constexpr可以用于修饰变量和函数,用于修饰变量需要保证变量的值在编译期间就能确定,用于修饰函数,要求函数的参数和返回值都是constexpr类型。 在大部分实际场景中,constconstexpr是可以混用的。比如可以把一个值声明为const常量并在声明的同constexpr进行初始化,例如: const int a = 5; constexpr int b = 5 * 4; 这样就既定义了一个不可改变的常量a,又使用constexpr编译期间计算出了一个常量b。 总结来说,constconstexpr都是用来定义常量的关键字,const声明一个不可改变的常量,而constexpr编译期间就可以计算出其值的常量。它们可以在大多数场景中混用。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *2* *3* [【C++】constconstexpr详解](https://blog.csdn.net/lemonxiaoxiao/article/details/128344695)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_1"}}] [.reference_item style="max-width: 100%"] [ .reference_list ]
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值