[C++读书笔记]常量表达式constexpr

文章目录

  • 常量表达式
  • constexpr变量
  • constexpr函数
  • 为何要使用constexpr

常量表达式

概念:

常量表达式是指值不会改变并且在编译过程就能得到计算结果的表达式

const int max_files = 20;			//是常量表达式
const int limit = max_files + 1;	//是常量表达式
int staff_size = 27;				//不是常量表达式
const int sz = get_size();			//不是常量表达式

解释:

对于staff_size,虽然他的初始值是字面值常量,但是他只是普通的int,而非const int,所以并不属于常量表达式

对于sz,虽然他是const int,但是除非get_size()函数不是一个常量表达式,否则他需要在运行过程中才能确定值,所以不属于常量表达式

!但是实际场景中,我们很难分辨一个初始值是不是常量表达式,那么又该怎么办呢?!

constexpr变量

概念:

这个变量是C++11引入的关键字,其作用是允许将变量(或函数)声明为constexpr类型以便让编译器来检查这是不是一个常量表达式(这就是最主要的作用),如果不满足则报错。

constexpr int mf = 20;				//是常量表达式
constexpr int limit = mf + 1;		//是常量表达式
constexpr int sz = size();			//只有当size是一个constexpr函数时才不会报错

int tmp = 10;
constexpr int val = tmp + 1;		//不是常量表达式,报错
    
const int tmp = 10;
constexpr int val = tmp + 1;		//是常量表达式

constexpr引用和指针

constexpr改变的是顶层const的属性(给变量添加了顶层const),而非底层const。

指针

在g++和vs下做了如下测试:

const int constval = 0;			//全局变量
int noconstval = 0;				//全局变量

int main()
{
	constexpr int const* p = &constval;
    //constexpr int *p = &constval;			//报错
	//*p = 10;								//不可修改,报错
    
	constexpr int* q = &noconstval;
	*q = 10;								//可修改
    
	return 0;
}

constexpr int *p = &constval;报错是因为p是顶层const,而constval是底层const,所以赋值失败。

*p = 10;报错是因为加了const后的p同时具有了顶层和底层const的属性,而底层const不能修改其指向变量的值。

引用:

引用与指针一样,也会有顶层const和底层const的限制

g++

static const int myStatic=42;
constexpr int const& myConstexprRef=myStatic;		//编译通过,但是不可修改myConstexprRef的值

static int myStatic=42;
constexpr int& myConstexprRef=myStatic;				//编译通过

vs

static const int myStatic=42;
constexpr int const& myConstexprRef=myStatic;		//编译通过,但是不可修改myConstexprRef的值

static int myStatic=42;
constexpr int& myConstexprRef=myStatic;				//编译通过

constexpr 引用和指针相似,并不保证它引用的对象是不可修改的;它只保证在引用创建时,所引用的对象是一个常量表达式。如果你能够编译并运行修改 myStatic 的代码,那么 myStatic 的值是可以被修改的,即使它通过一个 constexpr 引用被引用。

其实这种做法本质上是不合法的,因为引用的目的就是让这两个值绑定,而constexpr的目的是让值在程序于编译期间就确定值,然后运行期间持续用该值,不应产生改变。但是由于被绑定对象没有声明const或constexpr,从而导致了该值可以被修改。这种代码在严格遵循C++标准的编译器上是不能运行的,但是一些编译器出于多种目的,会允许这种做法。

总结:

被constexpr修饰的变量本身是不能修改的,但是它指向/引用的非const/非constexpr对象则可以被修改
所以如果你不想指向/引用的对象可以被修改,要么给该对象加上const,要么给该对象加上constexpr

constexpr函数

概念:

constexpr函数是指能用于常量表达式的函数。需要遵循几项约定:

  1. 返回类型和所有形参类型都要是字面值类型
  2. 只能有一条return语句

以下2个函数都是constexpr函数

constexpr int new_sz()
{
    return 0;
}
//对其的调用
constexpr int ret = new_sz();	  //常量表达式

constexpr size_t scale(size_t cnt)//当cnt是常量表达式时,scale(cnt)才是常量表达式
{
    return new_sz()*cnt;
}
//对其的调用
constexpr int ret = scale(1);	 //常量表达式

当一个函数被声明为constexpr时,它就会被隐式地指定为内联函数。

切记:constexpr函数不一定返回常量表达式!如:

int i = 10;
int ret = scale(i);						//返回的不是常量表达式
//constexpr int ret = scale(i);			//不能用constexpr接收,会报错

constexpr int ret = scale(10);	

为何要使用constexpr

  1. 编译时求值:

    在编译时进行求值,避免了运行时的计算开销,提高了程序的性能和效率

  2. 宏替代:

    使用constexpr可以取代宏,避免了宏带来的一些问题,如类型安全和可读性

    /*示例1*/
    #define PI 3.14159
    constexpr double PI = 3.14159;
    
    /*示例2*/
    #define MAX(a, b) ((a) > (b) ? (a) : (b))
    constexpr int max(int a, int b) 
    {  
        return (a > b) ? a : b;  
    }  
    constexpr int MAX_VALUE = max(10, 20);
    
  3. 模板元编程

    使用constexpr代替宏,对于模板元编程来说提供了更加清晰的语法和更强的类型检查

    /*示例3*/
    template<int N>  
    struct Factorial {  
        static const int value = N * Factorial<N - 1>::value;  
    };  
      
    template<>  
    struct Factorial<0> {  
        static const int value = 1;  
    };  
      
    #define FACTORIAL_5 Factorial<5>::value
    //-------------------------------------------------//
    template<int N>  
    constexpr int factorial() {  
        return N * factorial<N - 1>();  
    }  
      
    template<>  
    constexpr int factorial<0>() {  
        return 1;  
    }  
      
    constexpr int FACTORIAL_5 = factorial<5>();
    

参考文献:
《C++Primer中文第5版》

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mfbz.cn/a/455966.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

C++第四弹---类与对象(一)

✨个人主页&#xff1a; 熬夜学编程的小林 &#x1f497;系列专栏&#xff1a; 【C语言详解】 【数据结构详解】【C详解】 类与对象 1、面向过程和面向对象初步认识 2、类的引入 3、类的定义 4、类的访问限定符及封装 4.1、访问限定符 4.2、封装 5、类的作用域 6、类的…

WebP格式图像:起源、优势、兼容性及在线压缩方法

关于作者&#xff1a; 还是大剑师兰特&#xff1a;曾是美国某知名大学计算机专业研究生&#xff0c;现为航空航海领域高级前端工程师&#xff1b;CSDN知名博主&#xff0c;GIS领域优质创作者&#xff0c;深耕openlayers、leaflet、mapbox、cesium&#xff0c;canvas&#xff0…

第四百回 channel

文章目录 1. 知识回顾2. 示例代码3. 经验总结 我们在上一章回中介绍了MethodChannel的使用方法&#xff0c;本章回中将介绍EventChannel的使用方法.闲话休提&#xff0c;让我们一起Talk Flutter吧。 1. 知识回顾 我们在前面章回中介绍了通道的概念和作用&#xff0c;并且提到了…

数字生活的未来:探索Web3的全新世界

随着科技的飞速发展&#xff0c;我们正迈向一个数字化的未来。而在这个数字化的时代&#xff0c;Web3技术的崛起正引领着我们进入一个全新的世界。本文将深入探讨Web3技术的特点以及它给我们带来的全新体验。 1. 去中心化的特点 Web3的去中心化是其最显著的特点之一&#xff0…

若依Cloud项目配合nacos进行多环境profile的配置

1、前言 最近做的用若依改造的一个项目要把里面的配置挪到nacos的配置中心&#xff0c;之前用过apollo做配置中心&#xff0c;nacos用的很少&#xff0c;而且是自己从头做的那种&#xff0c;而自己想要实现的效果是本地启动和到测试&#xff0c;预发&#xff0c;生产环境启动的…

如何在Linux Archcraft中配置SSH服务并结合内网穿透实现远程连接

文章目录 1. 本地SSH连接测试2. Archcraft安装Cpolar3. 配置 SSH公网地址4. 公网远程SSH连接小结 5. 固定SSH公网地址6. SSH固定地址连接 Archcraft是一个基于Arch Linux的Linux发行版&#xff0c;它使用最简主义的窗口管理器而不是功能齐全的桌面环境来提供图形化用户界面。 C…

标准砂轮加工麻花钻或者铣刀螺旋槽齿形

螺旋槽与砂轮的空间位置运动关系可用下图表示&#xff0c;螺旋槽的形成靠工件绕轴线的旋转以及砂轮沿轴线的移动来完成的&#xff0c;以下坐标系可以清楚的描述二者之间的运动关系&#xff1a; 砂轮的形状如下&#xff1a; 经过坐标变换和下面这个重要的公式来计算工件的齿形…

JAVA初阶数据结构(链表)练习(这些可以作为java包中的方法)

这里的每一个题大家都要仔细完成&#xff0c;这些题目每个我都至少思考了两个小时左右&#xff08;沉重心&#xff0c;慢慢来&#xff09; 1.反向链表的实现&#xff08;对链表进行翻转&#xff09;&#xff08;力扣有&#xff09; &#xff08;1&#xff09;图示 &#xff0…

MADQN:多代理合作强化学习

处理单一任务是强化学习的基础&#xff0c;它的目标是在不确定的环境中采取最佳行动&#xff0c;产生相对于任务的最大长期回报。但是在多代理强化学习中&#xff0c;因为存在多个代理&#xff0c;所以代理之间的关系可以是合作的&#xff0c;也可以是对抗&#xff0c;或者两者…

java组合模式揭秘:如何构建可扩展的树形结构

组合模式&#xff08;Composite Pattern&#xff09;是一种结构型设计模式&#xff0c;它允许将对象组合成树形结构以表示整体/部分层次结构。组合模式使得客户端可以统一对待单个对象和组合对象&#xff0c;从而使得客户端可以处理更复杂的结构。 组合模式的主要组成部分包括&…

C#构造函数

C#中的构造函数是一种特殊的方法&#xff0c;用于创建和初始化类的对象。构造函数的名称与类的名称相同&#xff0c;并且没有返回类型。 在C#中&#xff0c;构造函数有以下几种类型&#xff1a; 默认构造函数&#xff1a;如果在类中没有定义构造函数&#xff0c;系统将自动提供…

基于java+springboot+vue实现的小区物业管理系统(文末源码+Lw+ppt)23-34

摘 要 随着互联网时代的发展&#xff0c;传统的线下管理技术已无法高效、便捷的管理信息。为了迎合时代需求&#xff0c;优化管理效率&#xff0c;各种各样的管理系统应运而生&#xff0c;在人们生活环境要求不断提高的前提下&#xff0c;小区物业管理系统建设也逐渐进入了…

2001-2022年上市公司数字化转型程度指数测算数据(含原始数据+测算代码+计算结果)(无形资产衡量)

2001-2022年上市公司数字化转型程度指数测算数据&#xff08;含原始数据测算代码计算结果&#xff09; 1、时间&#xff1a;2001-2022年 2、指标&#xff1a;证券代码、证券简称、统计截止日期、是否发生ST或*ST或PT、、是否发生暂停上市、行业代码、行业名称、stkcd、year、…

【Python】成功解决NameError: name ‘plt‘ is not defined

【Python】成功解决NameError: name ‘plt’ is not defined &#x1f308; 个人主页&#xff1a;高斯小哥 &#x1f525; 高质量专栏&#xff1a;Matplotlib之旅&#xff1a;零基础精通数据可视化、Python基础【高质量合集】、PyTorch零基础入门教程&#x1f448; 希望得到您…

【PHP安全】PHP伪协议

PHP伪协议&#xff1a; file:// #访问本地文件系统http:// #访问HTTPs网址ftp:// #访问ftp URLphp:// #访问输入输出流zlib:// #压缩流data:// #数据&#xff08;RFC 2397&#xff09;ssh2:// #security shell2expect:// #处理交互式的流glob:// #查找匹配的文件路径phar:// #P…

【MySQL性能优化】- 一文了解MVCC机制

MySQL理解MVCC &#x1f604;生命不息&#xff0c;写作不止 &#x1f525; 继续踏上学习之路&#xff0c;学之分享笔记 &#x1f44a; 总有一天我也能像各位大佬一样 &#x1f3c6; 博客首页 怒放吧德德 To记录领地 &#x1f31d;分享学习心得&#xff0c;欢迎指正&#xff…

打卡学习kubernetes——kubernetes架构原理

接上一篇的内容&#xff0c;除了核心组件&#xff0c;还有一些推荐的Add-ons&#xff1a; kube-dns 负责为整个集群提供DNS服务Ingress Controller 为服务提供外网入口Heapster 提供资源监控&#xff08;没用过这个&#xff0c;但是用过grafana&#xff0c;很方便&#xf…

5.Python从入门到精通—Python 运算符

5.Python从入门到精通—Python 运算符 Python 运算符算术运算符比较&#xff08;关系&#xff09;运算符赋值运算符逻辑运算符位运算符成员运算符身份运算符运算符优先级 Python 运算符 Python语言支持以下类型的运算符: 算术运算符比较&#xff08;关系&#xff09;运算符赋…

基于YOLOv5的火灾检测系统

&#x1f4a1;&#x1f4a1;&#x1f4a1;本文主要内容:详细介绍了火灾检测整个过程&#xff0c;从数据集到训练模型到结果可视化分析。 博主简介 AI小怪兽&#xff0c;YOLO骨灰级玩家&#xff0c;1&#xff09;YOLOv5、v7、v8优化创新&#xff0c;轻松涨点和模型轻量化&#…

CSS 03

1.选择器 1.1 结构伪类选择器 代码&#xff1a; <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><meta name"viewport" content"widthdevice-width, initial-scale1.0"><title>结…