C++知识整理系列(三)—— constexpr常量表达式

article/2025/10/8 6:28:06
  • const修饰常量,但是const并未区分编译时常量和运行时常量,而constexpr则只能是编译时常量,在C++11中提出。
  • 这篇文章,将详细讲解constexpr。

目录

  • 一、常量表达式
  • 二、constexpr变量
    • 三、constexpr函数
    • 四、字面值类型
  • 五、指针和constexpr
  • 六、字面值常量类
  • 参考

一、常量表达式

常量表达式(const expression):指值不会改变并且在编译阶段过程就能得到计算结果的表达式。

以下两种是常量表达式:

const int maxSize = 10;
const int limit = maxSize + 1;

以下两种不是常量表达式:

int staff_size = 27;
const int sz = get_size();
  • staff_size的初始值虽然是个字面值常量,但它的数据类型只是普通的int而非const int,还是可以被重新赋值的,所以不是常量表达式。
  • sz本身是一个常量,但它的具体值直到运行时才能获取到,所以也不是常量表达式。

二、constexpr变量

在一个复杂系统中,很难分辨一个初始值到底是不是常量表达式。从前面的例子可以发现,即使变量加上const,但是赋值是在运行时确定的也不是常量表达式。

C++11新标准规定,允许将变量声明为constexpr类型以便由编译器来验证变量的值是否是一个常量表达式。

  • 声明为constexpr的变量一定是一个常量。
  • 必须用常量表达式初始化。
constexpr int mf = 20;					//20是常量表达式
constexpr int limit = mf + 1;			//mf + 1是常量表达式
constexpr int sz = size();				//只有当size是一个constexpr函数时才是一条正确的声明语句

size()函数也需要constexpr修饰,成为constexpr函数。

三、constexpr函数

constexpr函数指能用于常量表达式的函数。定义constexpr函数有几项约定:

  • 函数的返回值类型及所有的类型都得是字面值类型
  • 函数体中必须有且只有一条return语句。
constexpr int new_sz() { return 40; }
constexpr int foo = new_sz();		//正确:foo是一个常量表达式

因为编译器能在程序编译时验证new_sz函数返回的是常量表达式,所以可以用new_sz函数初始化constexpr类型的变量foo。

(1)执行初始化任务时,编译器把对constexpr函数的调用替换成其结果值。为了能在编译过程中随时展开,constexpr函数被隐式地指定为内联函数

(2)constexpr函数体内也可以包含其他语句,只要这些语句在运行时不执行任何操作就行。例如,constexpr函数中可以有空语句、类型别名、using声明。

(3)constexpr函数的返回值可以不是一个常量

//cnt如果是常量表达式,返回值就是常量表达式
constexpr size_t scale(size_t cnt) { return new_sz() * cnt; }

比如,下面两个例子:

int arr[scale(2)];			//正确:scale(2)是常量表达式
int i = 3;
int a[scale(i)];			//错误:scale(i)不是常量表达式
  • 给scale传入字面值为2的常量表达式时,它的返回类型也是常量表达式。此时编译器用对应的结果值(80)替换为对scale函数的调用。
  • 当我们用一个非常量表达式调用scale函数时,比如int i = 3的对象i,返回值则不是一个常量表达式。当把scale函数用在需要常量表达式的上下文中时,编译器发现不是常量表达式,发出错误信息。

(4)constexpr函数通常定义在头文件中。因为编译器要想展开函数不仅需要函数声明还需要函数定义,而constexpr函数可以在程序中多次定义,但多个定义必须完全一致。

四、字面值类型

常量表达式的值需要在编译时就得到计算,因此对声明constexpr时用到的类型必须有所限制。因为这些类型一般比较简单,值也显而易见、容易得到,称之为"字面值类型"(literal type)。

字面值类型包括:算数类型、引用、指针,自定义类、string等类型不是字面值类型,也就不能定义成constexpr。

尽管指针和引用都能定义成constexpr,但它们的初始值却受到严格限制。一个constexpr指针的初始值必须是nullptr或者0,或者是存储在某个固定地址中的对象。

函数体内定义的变量一般来说并非存放在固定地址中,因此constexpr指针不能指向这样的变量。相反的,定义在函数体之外的对象地址固定不变,能用来初始化constexpr指针。

五、指针和constexpr

(1)如果在constexpr声明中定义了一个指针,限定符constexpr仅对指针有效,与指针所指的对象无关。

const int *p = nullptr;					//p是一个指向整数常量的指针
constexpr int *q = nullptr;				//q是一个指向整数的常量指针

q是一个常量指针,因为constexpr把它所定义的对象置为了顶层const。类似于:int *const q = nullptr;

(2)与其他常量指针类似,constexpr指针即可以指向常量也可以指向一个非常量:

constexpr int *np = nullptr;			//np是一个指向整数的常量指针,其值为空
int j = 0;
constexpr int i = 40;					//i的类型是整数常量
//假设i和j都定义在函数体之外
constexpr const int *p = &i;			//p是常量指针,指向整型常量i
constexpr int *p1 = &j;					//p1是常量指针,指向整数j

六、字面值常量类

constexpr函数的参数和返回值必须是字面值类型。注意,函数的返回值必须是字面值类型,但可以不是一个常量。

和其他类不同,字面值类型的类可能含有constexpr函数成员。这样的成员必须符合constexpr函数的所有要求,它们是隐式const。

字面值常量类:数据成员都是字面值类型的聚合类。如果一个类不是聚合类,但它符合下述要求,则它也是一个字面值常量类:

  • 数据成员都必须是字面值类型。
  • 类必须至少含有一个constexpr构造函数。
  • 如果一个数据成员含有类内初始值,这内置类型成员的初始值必须是一条常量表达式;如果成员属于某种类类型,这初始值必须使用成员自己的constexpr构造函数。
  • 类必须使用析构函数的默认定义,该成员负责销毁类的对象。

尽管构造函数不能是const的,但是字面值常量类的构造函数可以是constexpr函数。一个字面值常量类必须至少提供一个constexpr构造函数。

参考

  • C++ Primer


码字不易,觉得不错的小伙伴可以一键三连支持一下~
在这里插入图片描述


http://chatgpt.dhexx.cn/article/dgDIomU7.shtml

相关文章

C语言中的问号表达式

这道题目中的w<x?w:z<y?z:x 可以写为&#xff1a;w<x?w:(z<y?z:x) 首先z<y真的所以根据口诀真前假后得**(z<y?z:x)z1** 又因为w<x为假&#xff0c;所以得整个表达式为&#xff1a;1

正则表达式-问号(?)的用法

世界上并没有完美的程序&#xff0c;但是我们并不因此而沮丧&#xff0c;因为写程序就是一个不断追求完美的过程。 关于正则表达式&#xff0c;语法简单&#xff0c;但是要完成一个自己想要的逻辑判断就不那么容易了&#xff0c;今天主要讲解问号的一种用法&#xff0c;大家都知…

TypeScript 中问号+点 (?.) 和双问号 (??) 的含义

问号点表达式 (可选链操作符&#xff1a;?.) 例如 let res obj?.arr?.length 等价于 let res obj && obj.arr && obj.arr.length 如果不加问号&#xff0c;光是 obj.arr.length&#xff0c;可能报错没有 length 这个属性&#xff0c;如图&#xff1a; 因…

正则表达式中问号的用法

1、需求背景 由于最近工作需要&#xff0c;要从网页链接中找到网页中有用的博客内容&#xff0c;大家都知道&#xff0c;基本使用正则表达式来匹配是最简单的一种做法&#xff0c;而一般都是div中有div&#xff0c;怎么才能匹配到那些内容的div而不是一直匹配到最后面的div呢?…

问号表达式与逗号表达式问题

问号表达式&#xff1a; #include<stdio.h> void main() {int a,b,c,d,e;c9;d8;scanf("%d,%d",&a,&b);e(a>b)?c:d;printf("%d\n",e); } 可以理解为&#xff0c;如果a>b则ec&#xff0c;否则ed。“”后整体表达式可以看做&#xff08…

JavaScript 中问号的三种用法 ??和?.以及?: 您知道吗?

最近看了一些关于JavaScript的测试脚本&#xff0c;觉得JS 中问号的用法还是蛮有意思的&#xff0c;于是做了一下总结&#xff0c;在这里分享给大家&#xff01;JS中的问号大概有三种用法&#xff0c;分别是&#xff1a;空值合并操作符、可选链操作符和三目运算。 问号问号&…

实验6:shell编程

实验目的 &#xff08;1&#xff09;掌握shell编程的变量、程序控制结构、条件测试等语法。 &#xff08;2&#xff09;了解shell 脚本的运行方式 &#xff08;3&#xff09;理解shell脚本中的函数 &#xff08;4&#xff09;掌握简单shell脚本编写方法实验环境 &#xff08;1&…

shell编程练习题

近来学习到linux的shell编程一块&#xff0c;于是学完后想练习一下。本文就是基于一道练习题来记录一下自己遇到的问题以及解决的过程。 首先看题目&#xff1a;编写shell脚本&#xff0c;要求实现如下功能&#xff1a;当执行一个程序的时候&#xff0c;这个程序会让使用者选择…

Shell编程三剑客

文章目录 前言grep选项 sed 工具概述基本语法常见的 sed 命令选项常见的操作 实验结合正则表达式输出符合条件的文本删除符合条件的文本替换符合条件的文本迁移符合条件的文本 使用脚本编辑文件调用变量来改文件 awk概述工作原理&#xff1a;选项工作原理内置变量关于数组与字符…

Shell编程【万字Shell详细介绍带你入门建议收藏】

文章目录 1 Shell 基础介绍1.1 Shell 简介Shell 概述Shell 发展史查看Shell查看系统默认安装的 Shell查看当前登录用户默认 Shell查看当前的 Shell 1.2 Shell 脚本(定义、作用、格式、权限及执行)Shell 脚本基础知识Shell 脚本的约束Shell 脚本可以完成很多任务&#xff0c;但不…

Linux之Shell编程详解

精心整理Shell编程的入门&#xff0c;并配图加代码&#xff0c;方便大家跟着操作&#xff0c;但是难免不了存在纰漏&#xff0c;感谢大家的指正与理解&#xff01;觉的写的不错的小伙伴儿&#xff0c;一键三连支持一下&#xff0c;后期会有持续更新&#xff01;&#xff01;谢谢…

Linux系统的Shell编程

一. 什么是Shell 1. 在学习Shell编程之前&#xff0c;我们应该要先知道什么是Shell 用户、Shell、Linux内核、硬件的关系如下图&#xff1a; Shell是一个应用程序&#xff0c;也可以说是一个命令解释器&#xff0c;它是用户和Linux内核之间的桥梁&#xff0c;可以将用户在图形界…

Linux中的Shell编程

1.Shell的概念 shell是一个命令行解释器&#xff0c;它为客户提供了一个Linux内核发送请求一边运行程序界面系统级程序&#xff0c;用汉语可以通过shell启动、挂起、停止甚至编写一些程序。 shell还是一个功能强大的编程语言&#xff0c;易于编辑&#xff0c;易于调试&#xff…

Shell编程实验

实验二 Shell编程 文章目录 实验二 Shell编程一、如果当前目录下有文件f1&#xff0c;但是没有f2&#xff0c;解释命令ls f1 f2 2>ef1 1>&2的运行结果。二、使用for循环语句编写一段B-shell程序&#xff0c;完成显示用户注册目录下的a_sub, b_sub子目录下的所有C程序…

Shell编程入门

文章目录 1 Shell编程简介2 Shell脚本的执行方式3 Shell变量3.1 Shell变量介绍3.2 定义变量的规则3.3 将命令的返回值赋给变量&#xff08;重点&#xff09; 4 设置环境变量5 位置参数变量6 预定义变量7 运算符8 条件判断8.1 基本介绍8.2 if 判断8.3 case 语句 9 循环语句9.1 f…

Linux Shell编程

Linux Shell编程 一、简述 简单的介绍shell脚本的基本用法。 shell脚本是 由一些按照一定格式组合起来的shell命令 组成。shell脚本不需要编译就可以直接执行&#xff0c;它是边解释边执行的。 二、命令解释器 Linux系统提供多种不同的Shell以供选择。常用的有Bourne Shell&a…

Shell编程入门学习

文章目录 Shell编程Shell脚本的执行方式Shell的变量Shell的变量的介绍shell变量的定义 设置环境变量位置参数变量介绍基本语法 预定义变量基本介绍基本语法 运算符基本语法 条件判断判断语句常用判断条件 流程控制if 判断case语句for循环while循环 read读取控制台输入基本语法 …

shell编程

文章目录 一、shell简介二、shell脚本的执行方式三、shell变量3.1 shell变量介绍3.2 shell变量的定义3.2.1 基本语法3.2.2 定义变量的规则3.2.3 将命令的返回值赋予变量 四、环境变量的设置五、位置参数变量5.1 基本介绍5.2 基本语法 六、预定义变量6.1 基本介绍6.2 基本语法 七…

shell编程入门(一天掌握shell编程)

一、背景 到新公司之后&#xff0c;发现好多地方需要使用shell来编写一些简单的程序来提升自己的工作效率&#xff0c;因此专门B站上查看视频&#xff08;尚硅谷的视频&#xff09;&#xff0c;在这里学习总结下shell编程。 二、shell概述 shell是一个命令行解释器&#xff…

Linux【实操篇】—— Shell 编程入门、变量、运算符、条件判断、流程控制

目录 一、Shell 编程入门 1. 认识 Shell 2. Shell 脚本的创建与执行 二、Shell 变量 1. 系统变量和自定义变量 2. 变量的基本规则 3. 设置环境变量 4. 位置参数变量 5. 预定义变量 三、运算符 四、条件判断 五、流程控制 1. if 语句 2. case 语句 3. fo…