当前位置:首页 > 单片机 > CPP开发者
[导读]C20新增了两个const相关的关键字,于是当前存在四个相似的关键字:const,constexpr,consteval和constinit。接下来分别来进行讨论。第一,经过const修饰的变量具有只读属性,并且初始化发生于运行期。也就是说,若一个变量定义之后不允许被修改,就应该...

C 20新增了两个const相关的关键字,于是当前存在四个相似的关键字:const,constexpr,consteval和constinit。接下来分别来进行讨论。

第一,经过const修饰的变量具有只读属性,并且初始化发生于运行期。也就是说,若一个变量定义之后不允许被修改,就应该给它加上const。若在一个成员函数中不修改任何成员变量,就应该在成员函数后面加上const。

但是,它也可能发生于编译期,例如以const int代替宏来定义数组大小。

第二,经过constexpr修饰的变量或是函数,既保证只读,又发生于编译期。

然而,只有在参数是常量,和显式地以其返回值来初始化一个编译期常量时,它修饰的函数才会一定发生于编译期。如:

1#include 
2
3constexpr int sqr(int n) {
4    return n * n;
5}
6
7
8int main() {
9
10  // compile time
11  static_assert(sqr(10) == 100);
12
13  // compile time
14  int array[sqr(10)];
15
16  // compile time
17  constexpr int res = sqr(10);
18
19  // compile time or run time
20  int tmp = sqr(10);
21
22  // run time
23  int a = 10;
24  int tmp2 = sqr(a);
25}
此处,最后两个都可能发生于运行期。

第三,consteval用于创建一个immediate function(立即函数),immediate function的每次调用都会创建一个编译期常量。

经过该关键字修饰的函数会自动inline,这个函数中不能包含static数据,或是try、goto、new这种指令,也不能调用或使用非常量的函数与数据。

所以,简单来说,经过consteval修饰的函数必定会在编译期解析。

一个小例子:

1#include 
2
3int sqr1(int n) {
4    return n * n;
5}
6
7constexpr int sqr2(int n) {
8    return n * n;
9}
10
11consteval int sqr3(int n) {
12    return n * n;
13}
14
15
16int main() {
17
18  //constexpr int res1 = sqr1(10); // error! run time
19  constexpr int res2 = sqr2(10); // compile time
20  constexpr int res3 = sqr3(10); // compile time
21
22  int a = 10;
23  int res4 = sqr1(a); // run time
24  int res5 = sqr2(a); // run time
25  //int res6 = sqr3(a); // error! compile time
26
27}
这里有三个版本的sqr函数,sqr1()无任何修饰,所以只能在运行期调用;sqr2()以constexpr修饰,可能发生于运行期,也可能发生于编译期,取决于传入的参数;sqr3()以consteval修饰,所以必定发生于编译期。

因此,企图用运行期函数(sqr1)的返回值来初始化编译期常量,或是企图用非常量作为参数去调用编译期函数(sqr3),都将以失败告终。

第四,constinit可以确保变量初始化于编译期,这些变量须得处于静态存储区或是线程存储期。

静态存储区的变量,指的是全局变量、static变量或static成员变量;线程存储区的变量,指的是以thread_local修饰的变量,就是和线程生命期绑定的局部变量。

举个例子:

1#include 
2
3constexpr int val1 = 100;
4constinit int val2 = 100;
5
6
7int main() {
8
9  //std::cout << " val1 " <<  val1 << '\n'; // error
10  std::cout << " val2 " <<  val2 << '\n';
11
12  constexpr auto val3 = 100;
13  //constinit auto val4 = 100; // error
14  constinit thread_local auto val5 = 100;
15}
如第13行所示,不能用constinit修饰非静态的局部变量。

以constexpr和constinit修饰的变量都存在于编译期,不同之处在于,constexpr带有只读属性,而constinit没有。

最后,画张图来总结一下。


这里有两个维度,横向是运行时期,纵向是读写权限。关键字的作用就是来限定变量和函数在这两个维度的表现。

由于四个关键字不是同一时间引入的,所以const和constexpr有交叉部分(图中没体现)。就是说,const也可以表现编译期,constexpr也可以表现运行期。

而C 20新加的两个关键字,分别用于限定非只读的编译期变量和函数,变量必须存在于静态存储区或线程存储区,函数必须是immediate function。


- EOF -

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: 驱动电源

在工业自动化蓬勃发展的当下,工业电机作为核心动力设备,其驱动电源的性能直接关系到整个系统的稳定性和可靠性。其中,反电动势抑制与过流保护是驱动电源设计中至关重要的两个环节,集成化方案的设计成为提升电机驱动性能的关键。

关键字: 工业电机 驱动电源

LED 驱动电源作为 LED 照明系统的 “心脏”,其稳定性直接决定了整个照明设备的使用寿命。然而,在实际应用中,LED 驱动电源易损坏的问题却十分常见,不仅增加了维护成本,还影响了用户体验。要解决这一问题,需从设计、生...

关键字: 驱动电源 照明系统 散热

根据LED驱动电源的公式,电感内电流波动大小和电感值成反比,输出纹波和输出电容值成反比。所以加大电感值和输出电容值可以减小纹波。

关键字: LED 设计 驱动电源

电动汽车(EV)作为新能源汽车的重要代表,正逐渐成为全球汽车产业的重要发展方向。电动汽车的核心技术之一是电机驱动控制系统,而绝缘栅双极型晶体管(IGBT)作为电机驱动系统中的关键元件,其性能直接影响到电动汽车的动力性能和...

关键字: 电动汽车 新能源 驱动电源

在现代城市建设中,街道及停车场照明作为基础设施的重要组成部分,其质量和效率直接关系到城市的公共安全、居民生活质量和能源利用效率。随着科技的进步,高亮度白光发光二极管(LED)因其独特的优势逐渐取代传统光源,成为大功率区域...

关键字: 发光二极管 驱动电源 LED

LED通用照明设计工程师会遇到许多挑战,如功率密度、功率因数校正(PFC)、空间受限和可靠性等。

关键字: LED 驱动电源 功率因数校正

在LED照明技术日益普及的今天,LED驱动电源的电磁干扰(EMI)问题成为了一个不可忽视的挑战。电磁干扰不仅会影响LED灯具的正常工作,还可能对周围电子设备造成不利影响,甚至引发系统故障。因此,采取有效的硬件措施来解决L...

关键字: LED照明技术 电磁干扰 驱动电源

开关电源具有效率高的特性,而且开关电源的变压器体积比串联稳压型电源的要小得多,电源电路比较整洁,整机重量也有所下降,所以,现在的LED驱动电源

关键字: LED 驱动电源 开关电源

LED驱动电源是把电源供应转换为特定的电压电流以驱动LED发光的电压转换器,通常情况下:LED驱动电源的输入包括高压工频交流(即市电)、低压直流、高压直流、低压高频交流(如电子变压器的输出)等。

关键字: LED 隧道灯 驱动电源
关闭