广告
返回顶部
首页 > 资讯 > 后端开发 > 其他教程 >浅谈C++中const与constexpr的区别
  • 919
分享到

浅谈C++中const与constexpr的区别

2024-04-02 19:04:59 919人浏览 薄情痞子
摘要

目录一.const常量与#define比较二.const修饰1.修饰普通变量,必须初始化2.修饰类变量和成员变量3.修饰成员函数4.修饰指针5.修饰引用三.const转换四.顶层co

一.const常量与#define比较

define只是简单的替换,没有类型,const可以做到防窜改与类型安全

而且#define会在内存中可能(有几次替换就有几次拷贝)有多份拷贝,对于字面值常量加不加const都一样,例如:const char* arr = “123”;,储存在常量区,只有一份拷贝;对于局部对象,常量存放在栈区,例如:void add(){const char crr[] = “123”;},这里“123”本应储存在栈上,但编译器可能会做某些优化,将其放入常量区;对于全局对象,常量存放在全局/静态存储区;用const会比#define使用更少的空间,效率更高。

这里有一个小例子:char* brr = "123"; char drr[] = "123";前者字符串123存在常量区,不能通过brr去修改"123"的值;后者"123"保存在栈区,可以通过drr去修改。

现在C++除了一些特定用法,推荐用const,inline,enum等替换宏——来自《Effective C++》条款02

二.const修饰

1.修饰普通变量,必须初始化

const int a = 10; 表示int对象a,是一个常量,不可以改变值,从编译器生成二进制角度看,生成的a存放在.rodata段,也就是只读(readonly)区域。不过并不绝对,有的时间统计优化等级开的高,也不取地址,可能会优化成立即数在.text段中。

2.修饰类变量和成员变量


class cAAA{
public:
    cAAA(int a) : m_iV(a){}
    const int GetValue() const {return m_iV;}
    void AddValueOneTime(){m_iChangeV++;}
private:
    const int m_iV;
public:
    mutable int m_iChangeV;
    static const int m_iStaticV;
};
static const int m_iStaticV = 1000;

const cAAA aa(100);
aa.GetValue();
aa.m_iChangeV++;
  • cAAA类成员m_iV是const变量,必须放到初始化列表中进行初始化,不能进行赋值。
  • 对于静态常成员,与普通静态成员类似,推荐放到类外.cpp中初始化。
  • aa只能调用const函数,如aa.GetValue(),不能调用非常成员函数aa.AddValueOneTime()。
  • 对于这种const对象,又想修改成员,可以在成员声明加上mutable,这样const对象aa也可以修改m_iChangeV,这种用法比较少。

3.修饰成员函数

  • 表示这个函数可以被const对象调用,也可以被普通对象调用,不会改变对象的成员,也就是说更像一种只读不写型的逻辑运算,所以有些人推荐类成员函数,可以都加上const。有一个小技巧,当const和non-const成员函数有着实质等价的实现时,令non-const版本调用const版本可避免代码重复;但反过来不行,const函数内部也必须只能调用const函数—— 《Effective C++》条款03
  • 有一点要注意,编译器强制实施bitwase constness,但编写程序时应该使用conceptual constness,解决编译器的bitwase constness属性就用到了上述的mutable。关于介绍bitwase constness的具体表现可以参考《Effective C++》条款03。

4.修饰指针


const char* p1;
char const *p2;
char* const p3;
const char* const p4;

对于初学者来说这大概是很难理解的一个知识点,怎么区分这四个呢?记住秘诀,直接从右向左读就一招制敌了。

  • p1是一个指针,指向char字符常量,表示p1所指对象内容不可以改,所指地址可以改。
  • p2同p1,写法不同,两者等价。
  • p3是一个常量,且是个指针,指向char字符,表示p3所指对象内容可以改,所指地址不可以改。
  • p4是一个常量,且是个指针,指向char字符常量,表示p4所指对象内容不可以改,且所指地址也不可以改。
  • 相对来说p1,p2是最常用传参或者返回值的手段。

5.修饰引用

修饰引用和对象差不多,对象内容不可以改变。作为函数参数传参数,不存在copy开销,这是比较推荐的写法,例如:拷贝构造函数,赋值构造,STL里用于比较的函数或者仿函数,详情请参阅《Effective C++》条款20。bool Less(const cAAA& left, const cAAA& right);


float dValue = 1.05f;
const int& a = dValue;

const int iTemp = dValue;
const int& a = iTemp;

因为常引用不能改变,这种情况下编译器会创建一个临时变量来处理隐式转换,我们实际是对临时变量进行了常引用。

三.const转换

一般来说,从T*转换到const T*是比较简单的,且编译器支持的隐式转换,也可以显示的用模板处理,例如我们简单写一下RemoveConst模板,最后用using化名一下。但从const T*T*就麻烦一些,推荐使用const_cast。


template <typename T>
struct RemoveConst{
    typedef T Type;
};

template <typename T>
struct RemoveConst<const T>{
    typedef T Type;
};

template <typename T>
using RCType = typename RemoveConst<T>::Type;

四.顶层const与底层const

  • 简单来说const修饰的对象本身不能改变就是顶层const,但如果是指针或者引用的对象不能改变,则称为底层const。
  • const int cV = 10; cV是顶层const,本身不能改变
  • char const *p2; p2是底层const,p2本身值可以改变,但所指内容不可以改变
  • char* const p3; p3是顶层const,p3的本身值不可以改变
  • const char* const p4; p4既是顶层const,又是底层const
  • 注:对于上述模板RCType是无法移除p2这种底层const,如果要移除,请用const_cast<T*>移除,但这种操作可能引起Crash或者未知风险

const char* pA = "sss";
char* pB = const_cast<char*>(pA);
auto pC = RCType<decltype(pA)>(pA);
std::cout << "type is the same: " << std::is_same<decltype(pB), decltype(pC)>::value << std::endl;
std::cout << "pB Type Name: " << typeid(pB).name() << "pc Type Name: " << typeid(pC).name() << std::endl;
//pB[0] = 'A';//error, Segmentation fault

五.C++11新引入的constexpr

这个关键字表示这是一个常量表达式,是一个编译期就可以确认的值,最常用于模板中,例如模板递归求值。

它可不只是变量,例如:


const int iSize1 = sizeof(int);
const int iSize2 = GetSize();

iSize1是个常量,编译期的,但iSize2就不一定,它虽然不能改变,但要到GetSize()执行结束,才能知道具体值,这与常量一般在编译期就知道的思想不符,解决这个问题的方法就是改为:constexpr int iSize2 = GetSize(); 这样要求GetSize()一定要能在编译期就算出值,下面几个例子中GetSizeError()就会编译失败。GetFibo函数,编译期就已经递归计算出值。


constexpr int GetSize(){
  return sizeof(int) + sizeof(double);
}

constexpr int GetSizeError(){
  return random();
}

constexpr int GetCalc(int N){
  return N <= 1 ? 1 : N * GetCalc(N - 1);
}

const int iSize1 = sizeof(int);
constexpr int iSize2 = GetSize();
//constexpr int iSize3() = GetSizeError();
constexpr int iSize4 = GetCalc(10);
std::cout << iSize1 << " " << iSize2 << " " << iSize4 <<std::endl;

当然我们可以用模板写GetCalc函数:


template <int N>
struct TCalc{
  static constexpr int iValue = N * TCalc<N-1>::iValue;
};

template <>
struct TCalc<1>{
  static constexpr int iValue = 1;
};
std::cout << TCalc<10>::iValue << std::endl;

以上就是浅谈C++中const与constexpr的区别的详细内容,更多关于C++  const与constexpr区别的资料请关注编程网其它相关文章!

--结束END--

本文标题: 浅谈C++中const与constexpr的区别

本文链接: https://www.lsjlt.com/news/158668.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 浅谈C++中const与constexpr的区别
    目录一.const常量与#define比较二.const修饰1.修饰普通变量,必须初始化2.修饰类变量和成员变量3.修饰成员函数4.修饰指针5.修饰引用三.const转换四.顶层co...
    99+
    2022-11-12
  • C++中const与constexpr的区别是什么
    这篇文章主要讲解了“C++中const与constexpr的区别是什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“C++中const与constexpr的区别是什么”吧!一.const常量...
    99+
    2023-06-21
  • 浅谈JS中var,let和const的区别
    目录区别1区别2区别3区别4区别5区别6区别7区别1 let和var用来声明变量,const用来声明常量。 变量就是赋值后可以改变它的值,常量就是赋值后就不能改变它的值。 当声明为对...
    99+
    2022-11-13
  • 浅谈Java中replace与replaceAll区别
    在Java中,replace和replaceAll都是用于替换字符串中的字符或字符串的方法,但它们之间有一些区别。1. 参数类型:-...
    99+
    2023-08-14
    Java
  • 浅谈springboot@Repository与@Mapper的区别
    目录1、@Repository2、@Mapper3、区别相同点:不同点:4、解决使用@mapper接口时,注入mapper爆红问题今天在用springboot整合mybatis时,m...
    99+
    2022-11-13
  • C++中const char*、char const*、char * const三者的区别
    目录一、const char *ptr;二、char const *ptr;三、char * const ptr;C/C++ 中关于以下三种定义: const char *...
    99+
    2022-11-12
  • 浅谈C语言中include""与include<>的区别
    新建控制台应用程序 Win32 Application和Win32 Console Application 都是工作在32位Windows环境的程序。其中: (1)Win32 Ap...
    99+
    2022-11-12
  • 浅谈C++中thread库join和detach的区别
    目录C++11中thread库join和detach的区别线程状态线程环境join和detach的区别总结C++11中thread库join和detach的区别 线程状态 在一个线程...
    99+
    2022-11-12
  • 浅谈JSP serverlet的区别与联系
    JSP(JavaServer Pages)和Servlet都是Java Web开发中的关键技术,用于构建动态Web应用程序。它们之间...
    99+
    2023-08-15
    jsp
  • 浅谈C语言中的sizeof()和strlen()的区别
    目录sizeof()strlen补一个注意事项:sizeof()和strlen()经常会被初学者混淆,但其中有有很大区别: sizeof() 1. sizeof()【操作数所占空间的...
    99+
    2022-11-13
  • C++和C中const的区别详解
    目录C中的constconst修饰局部变量const修饰全局变量const修饰的全局变量有外部链接属性const与指针C++中的constconst修饰普通全局变量const修饰普通...
    99+
    2022-11-12
  • c语言和c++语言中const修饰的变量区别浅析
    目录c:修饰全局变量:修饰局部变量:c++:修饰全局变量:修饰局部变量:总结:在c语言中:在c++语言中:总结c: 修饰全局变量: 用const修饰的全局变量是没有办法直接修改的,间...
    99+
    2022-11-13
  • 浅谈Android onTouchEvent 与 onInterceptTouchEvent的区别详解
    首先从字面意思理解两个词 onTouchEvent:触发触摸事件 onInterceptTouchEvent:触发拦截触摸事件 通过查看源代码及类继承关系 onIntercep...
    99+
    2022-06-06
    Android
  • 浅谈Mysql tinyint(1)与tinyint(4)的区别
    目录引言什么是tinyint(M)测试总结参考文献引言 借由本篇文章来探讨下在mysql数据库中数值类型tinyint(1)和tinyint(4) 有啥区别呢? 什么是tinyint(M) 先来了解下mysql...
    99+
    2023-03-20
    Mysql tinyint(1)与tinyint(4) Mysql tinyint(1) Mysql tinyint(4)
  • 浅谈Mysql tinyint(1)与tinyint(4)的区别
    目录引言什么是tinyint(M)测试总结参考文献引言 借由本篇文章来探讨下在Mysql数据库中数值类型tinyint(1)和tinyint(4) 有啥区别呢? 什么是ti...
    99+
    2023-03-20
    Mysql tinyint(1)与tinyint(4) Mysql tinyint(1) Mysql  tinyint(4)
  • 浅谈Spring中几个PostProcessor的区别与联系
    目录Spring几个PostProcessor的区别首先明确 Bean 的生命周期:查看 IOC 容器创建时的调用流程spring-postProcessor的执行时机BeanPos...
    99+
    2022-11-12
  • 浅谈js中Object.create()与new的具体实现与区别
    目录Object.create与new区别Object.create()原理new原理继承比较组合继承与寄生组合继承组合继承寄生组合继承Object.create与new区别 fun...
    99+
    2022-11-13
  • 浅谈C结构和C++结构之间的区别
    今天我们来看一下:C结构和C++结构之间,到底有什么不一样地方! 在C++中,struct和class完全相同,除了struct默认为公共可见性和class默认为私有可见性。&nbs...
    99+
    2022-11-12
  • C++和C中const的区别是什么
    本篇内容介绍了“C++和C中const的区别是什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!const,这个词字面意思为:常数。这就表示...
    99+
    2023-06-21
  • 浅谈Golang的new与make区别是什么
    目录newmake小结:区别:在go语言中,make和new都是内存的分配(堆上),但是make只用于slice、map以及channel的初始化(非零值);而new用于类型的内存分...
    99+
    2022-11-13
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作