news 2026/6/10 16:36:58

constexpr 和 explicit 在 C++ 中被提出的动机

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
constexpr 和 explicit 在 C++ 中被提出的动机

相关内容参考:C++中constexpr 与 explicit关键字使用详解


1.constexpr——“让编译器做更多事”

提出动机:提升性能,减少运行时开销

在 C++11 之前:

  • 只有const,但const 不保证编译期求值
  • 想要编译期常量,只能用#defineenum——都不够安全、不支持函数、不支持复杂类型

因此提出constexpr解决两个痛点:

痛点1:需要真正的编译期常量(strong compile-time constant)

例如:

constintN=foo(10);// 不能保证 foo(10) 在编译期运行intarr[N];// 可能无法作为数组大小

C++ 希望:如果值能在编译期确定,就应该在编译期确定。


痛点2:需要编译期可执行的函数(constexpr function)

例如:

doublearea(doubler){return3.14159*r*r;}constexprdoubleA=area(10);// C++03 之前不行

C++ 希望:

  • 数学函数
  • 小工具函数
  • 初始化对象
  • 甚至类构造函数

都能在编译期执行。


constexpr给 C++ 带来的核心能力

编译期执行函数(Compile-time evaluation)

极大提高运行效率 + 常量折叠 + 去掉运行时开销。

更强的类型安全

#define可靠得多。

用于模板元编程(metaprogramming)

constexpr让模板元编程更直观,不必写复杂的 TMP 结构体推导。

允许类的编译期构造

例如:

structVec3{doublex,y,z;constexprVec3(doublex,doubley,doublez):x(x),y(y),z(z){}};

③ 总结——constexpr 提出的根本动机

让编译器“成为运行时”,能提前计算任何能计算的东西,提高性能与安全性,增强 C++ 的表达能力。

一句话:

constexpr = 让 C++ 从运行时语言 → 编译期可计算语言


2.explicit——“拒绝隐式转换导致的灾难”

提出动机:防止隐式类型转换引发 bug

在 C++98 之前:

构造函数可以隐式调用,导致非常多的意外错误:

structVec{Vec(intx){}// 可隐式转换};Vec v=5;// ??? 编译器会自动调用 Vec(5)

问题:

  • 难以察觉的隐式转换
  • 编译器默默干了不想要的事
  • 操作符重载时尤其危险

如:

booloperator==(constVec&,constVec&);Vec v;boolt=(v==10);// 会被隐式转成 Vec(10)

这是非常危险的行为。


② explicit 的目标:禁止不安全的隐式构造

explicitVec(intx);

禁止:

Vec v=5;// 不再允许

保留:

Vecv(5);// 明确构造

③ explicit 的更高层动机(C++11 之后)

C++11 支持:

  • explicit用于转换运算符
  • explicit(false)/explicit(true)(C++20)

用于禁止 “过度智能” 的隐式行为。

例如:

explicitoperatorbool()const;

防止:

Vector v;if(v){}//intx=v+1;// 防止自动转 bool 再转 int 的怪行为

④ 总结——explicit 提出的根本动机

explicit 的目的就是阻止编译器做你没说过的隐式转换,避免隐式构造引发隐蔽 bug。

一句话:

explicit = 禁止偷偷摸摸的隐式转换,保护你不被语言陷阱坑死。


对比总结

关键字核心动机解决的问题
constexpr让编译期可以计算更多东西,提高性能与安全性编译期常量、constexpr 函数、constexpr 构造函数
explicit禁止隐式转换,减少意料之外的 bug防止构造函数隐式调用,防止隐式转换

3constvsconstexpr对比

C++ 中constconstexpr都与常量相关,但用途和能力差异非常大:

特性constconstexpr
定义声明一个值不可修改声明一个值或函数可以在编译期求值
是否保证编译期不保证,可能运行时初始化保证,如果满足条件,必须在编译期计算
适用对象变量、成员、函数返回值变量、成员、函数、构造函数、类常量
初始化可以运行时初始化必须用编译期常量初始化(C++14 起 relax,可有简单运行时计算)
函数const函数限制成员修改(方法级)constexpr函数在编译期求值,并且可用于常量表达式
数组长度const int N = 10; int arr[N];编译器可能支持,但不保证constexpr int N = 10; int arr[N];必定可用于编译期大小
优化运行时常量,编译器可优化编译期常量,可完全消除运行时开销

举例

constinta=5;// 运行时可变对象也可初始化为常量intarr1[a];// 在一些编译器中可行,但非标准保证constexprintb=5;// 编译期常量intarr2[b];// 标准保证可用

函数对比

constintsquare_const(intx){returnx*x;}// 运行时求值constexprintsquare_constexpr(intx){returnx*x;}// 编译期求值可能
  • square_constexpr(3)→ 编译期可求值
  • square_const(3)→ 编译期不保证,只能运行时求值

4explicit在模板类 / 复杂构造中的用法

explicit最初是为了防止隐式类型转换,但在模板类 / 泛型编程中尤其重要:


① 模板类单参数构造

template<typenameT>structWrapper{explicitWrapper(T val):value(val){}T value;};Wrapper<int>w1(5);// 明确构造Wrapper<int>w2=5;// 错误,禁止隐式转换

在模板类中,如果不加explicit,可能会导致意想不到的隐式类型转换,尤其当模板参数为复杂类型(比如矩阵、点云类型)时。


② 多参数模板构造 + 默认参数

template<typenameT>structPoint{explicitPoint(T x=0,T y=0,T z=0):x(x),y(y),z(z){}T x,y,z;};Point<int>p1(1,2,3);// 明确构造Point<int>p2;// 默认参数构造Point<int>p3={};// 禁止隐式列表初始化

explicit可以避免模板类在列表初始化 / 隐式转换时产生不安全行为。


explicit与转换运算符(C++11+)

C++11 引入了explicit operator,用于模板类中的类型安全转换:

template<typenameT>structVec{T x,y;explicitoperatorbool()const{returnx!=0||y!=0;}};Vec<int>v{0,0};if(v){}// 允许intn=v;// 禁止隐式转换
  • 模板类中尤其重要,因为模板类型可能变化,不加explicit很容易产生隐式转换错误。

④ 小结

  1. const vs constexpr

    • const= 不可修改(可能运行时求值)
    • constexpr= 编译期求值,可用于模板/数组/常量初始化
  2. explicit 在模板类中

    • 防止单参数模板构造产生隐式类型转换
    • 避免列表初始化 / 默认参数带来的隐式调用
    • 可用于转换运算符,增加类型安全性

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

PyTorch数据加载瓶颈诊断与性能优化实战

PyTorch数据加载瓶颈诊断与性能优化实战 【免费下载链接】pytorch-deep-learning Materials for the Learn PyTorch for Deep Learning: Zero to Mastery course. 项目地址: https://gitcode.com/GitHub_Trending/py/pytorch-deep-learning 深度学习训练过程中&#xff…

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

终极Markdown幻灯片制作指南:如何用Marp快速创建专业演示

终极Markdown幻灯片制作指南&#xff1a;如何用Marp快速创建专业演示 【免费下载链接】marp The site of classic Markdown presentation writer app 项目地址: https://gitcode.com/gh_mirrors/ma/marp 还在为制作幻灯片而烦恼吗&#xff1f;Marp作为一款创新的Markdow…

作者头像 李华
网站建设 2026/6/6 8:12:18

14、使用软件检测工具测试系统安全性

使用软件检测工具测试系统安全性 1. 系统安全检测工具概述 在保障系统安全方面,有许多工具可供使用。这里将重点介绍两款工具:nmap 和 Tripwire。nmap 用于主动检查系统端口的漏洞,而 Tripwire 则用于被动记录系统的所有活动,方便后续查看和比较。 2. 使用 nmap 扫描网络…

作者头像 李华
网站建设 2026/6/10 10:47:23

15、企业 Linux 互联网服务之 DNS 服务器配置与管理

企业 Linux 互联网服务之 DNS 服务器配置与管理 1. DNS 简介 在网络世界中,我们每天都会在浏览器里输入网址,比如 www.redhat.com ,然后就能轻松访问到对应的网站。这背后其实是域名系统(DNS)在默默工作,它负责将我们容易记住的域名转换为计算机能够理解的 IP 地址,…

作者头像 李华
网站建设 2026/6/10 16:19:48

DreamScene2动态桌面终极指南:5分钟打造专属Windows视觉盛宴

DreamScene2动态桌面终极指南&#xff1a;5分钟打造专属Windows视觉盛宴 【免费下载链接】DreamScene2 一个小而快并且功能强大的 Windows 动态桌面软件 项目地址: https://gitcode.com/gh_mirrors/dr/DreamScene2 还在为单调的静态桌面感到乏味吗&#xff1f;当Windows…

作者头像 李华
网站建设 2026/6/10 12:20:13

Arduino HID开发终极指南:打造专属输入设备的神器

Arduino HID开发终极指南&#xff1a;打造专属输入设备的神器 【免费下载链接】HID Bring enhanced HID functions to your Arduino! 项目地址: https://gitcode.com/gh_mirrors/hi/HID 还在为找不到合适的输入设备而烦恼吗&#xff1f;想要让Arduino变身专业的USB外设控…

作者头像 李华