news 2026/6/10 12:43:20

跟我学C++中级篇——对类const关键字的分析说明

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
跟我学C++中级篇——对类const关键字的分析说明

一、说明

前面多篇文章分析了常量的各种用法以及在新标准中的各种扩展。到目前为止,可以发现有几个长得差不多的关键字:从早期的const到C++11中的constexpr直到C++20中的consteval和constinit。这些长得类似的const有什么区别和联系呢?下面就对它们的具体应用进行分析和说明。

二、对比和分析

针对上面的这些长得差不多关键字,主要有以下的不同:

  1. const
    它是在C++中传统,const主要用来控制运行时对变量和表达式的只读性的控制。目的在于保护相关数据的安全性并进行接口的约束。其约束的变量可以根据时机在编译期或运行期进行求值
  2. constexpr
    C++11中引入的constexpr,用于在编译期的计算,也就是编译期求值,同时对变量和表达式的只读性控制。可以将运行期的计算转到编译期的计算和常量表达式的处理,当然,也支持运行期的处理
  3. consteval
    作为C++20引入的constexpr,主要用于立即函数处理,即它只能修饰函数,它修饰的函数及相关调用及相关求值都强制要求必须在编译期完成
  4. constinit
    constinit也是C++20引入的,它只能用于对静态或线程局部变量进行“常量初始化”(constant initialization)即编译期完成初始化。否则,会报编译错误。需要注意的是,它只能是初始化,完成后,这个变量是可以再次修改的即这个变量不是常量。

三、应用场景和限制分析

在了解上它们四个关键字的情况后,就可以根据其特点来确定其应用场景:

  1. 编译期计算
    需要在编译期计算的场景下,可以使用constexpr和consteval,这样,就可以把运行时的开销转移到编译期。另外在模板和元编程中这两个关键字也是经常使用的
  2. 接口约束和数据只读
    在与旧代码兼容以及需要接口的常量化控制,并且只是需要对数据常量化的处理时,可以考虑使用const。当然,很多情况下也可以使用constexpr
  3. 强制常量初始化
    在前面也提到过,在C++库中的全局(静态)变量调用时可能会产生初始化顺序的问题,从而产生意想不到的问题。而如果使用constinit则可以避免相关变量动态初始化,从而保证变量的依赖安全性。此外在全局配置处理、跨编译单元的变量以及线程局部存储等场景下都可以应用

虽然constinit无法限制变量的常量性,但它却也不允许联合使用const和constexpr来限制变量的常量性(只读性)。而const和constexpr则可以联合使用,不过是一种冗余的应用,没有什么实际意义。

四、例程

下面看一下具体的例程的对比:

//下面代码来自 cppreference//constexprconstexprintf();constexpr bool b1=noexcept(f());// false, undefined constexpr functionconstexprintf(){return0;}constexpr bool b2=noexcept(f());// true, f() is a constant expression//constevalconstevalintsqr(intn){returnn*n;}constexprintr=sqr(100);// OKintx=100;intr2=sqr(x);// Error: Call does not produce a constantconstevalintsqrsqr(intn){returnsqr(sqr(n));// Not a constant expression at this point, but OK}constexprintdblsqr(intn){return2*sqr(n);// Error: Enclosing function is not consteval// and sqr(n) is not a constant}//constinitconstchar*g(){return"dynamic initialization";}constexprconstchar*f(bool p){returnp?"constant initializer":g();}constinitconstchar*c=f(true);// OK// constinit const char* d = f(false); // error

代码很简单,重点是对比一下。下面再看一个综合的应用:

#include<mutex>#include<string>intBASE_V=0;class Demo{public:// constexprstaticconstexprdoublePI=3.14;// constevaltemplate<typename T>constevalstaticTratioPI(T t){returnt*PI;}// constexpr-运行和编译期均可用template<typename T>constexprstaticTless(T min,T max){returnmin<max;}// consttemplate<typename T>staticconstauto&get(constT&t){returnBASE_V>t?BASE_V:1;}};class ControlMsg{public:voidaddmsg(conststd::string&msg){std::lock_guardlock(mutex_);msg_.append(msg);}private:std::string msg_="";std::mutex mutex_;};// constinitconstinit ControlMsg msg;// 使用示例intmain(){// 编译时constexprdoublenPI=Demo::ratioPI(3.0);static_assert(nPI>Demo::PI&&nPI<5*Demo::PI);// 运行时autostatus=Demo::less(3,5);//常量初始化msg.addmsg("Application started");return0;}

五、总结

有对比才有差距,才好掌握这四个关键字各自的特点和应用的场景。在实际应用的过程中,要根据实际情况来有针对性使用这些关键字。特别需要注意C++标准的版本不同的情况下,应用方式的不同。这样才能够编写出安全、高效的C++代码。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/6/10 11:21:30

显存不足跑不动大模型?万物识别镜像显存优化技巧

显存不足跑不动大模型&#xff1f;万物识别镜像显存优化技巧核心提示&#xff1a;本文基于阿里开源的「万物识别-中文-通用领域」模型&#xff0c;针对在PyTorch 2.5环境下因显存不足导致大模型推理失败的问题&#xff0c;系统性地提出一套可落地的显存优化方案。不仅适用于本地…

作者头像 李华
网站建设 2026/6/9 23:55:37

快速验证创意:用VueDraggable一小时打造看板原型

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 快速生成一个任务看板原型&#xff0c;功能包括&#xff1a;1. 三列看板&#xff08;待办/进行中/已完成&#xff09; 2. 使用VueDraggable实现跨列拖拽 3. 卡片拖拽时有视觉反馈 …

作者头像 李华
网站建设 2026/6/10 11:18:56

TensorRT加速尝试:进一步压缩推理延迟

TensorRT加速尝试&#xff1a;进一步压缩推理延迟 万物识别-中文-通用领域 在当前AI应用快速落地的背景下&#xff0c;模型推理效率已成为决定产品体验的关键瓶颈。尤其是在移动端、边缘设备或高并发服务场景中&#xff0c;毫秒级的延迟优化都可能带来用户体验的显著提升。本文…

作者头像 李华
网站建设 2026/6/10 11:30:10

实战案例:电商商品自动分类系统3天上线全记录

实战案例&#xff1a;电商商品自动分类系统3天上线全记录 在电商行业&#xff0c;商品数量庞大、品类繁杂&#xff0c;人工分类成本高、效率低&#xff0c;且容易出错。随着AI技术的发展&#xff0c;利用图像识别实现自动化商品分类已成为提升运营效率的关键路径。本文将带你完…

作者头像 李华
网站建设 2026/6/10 11:22:03

1小时打造智能远程启动管理原型

快速体验 打开 InsCode(快马)平台 https://www.inscode.net输入框内输入如下内容&#xff1a; 快速开发一个远程启动管理系统的概念验证原型。核心功能包括&#xff1a;1) 设备发现与列表展示 2) 单个设备远程启动/停止 3) 基本状态反馈 4) 简易日志记录。使用Python Flask框…

作者头像 李华