当前位置:首页 > 公众号精选 > 混说Linux
[导读]其实在 c++语言里面const修饰的才算是一个真正的常量,在 c 语言中 const 可以说是个“冒牌货”。为什么会这样?其实是 c++ 编译器对 const 进行了加强,当 c++ 编译器遇到常量声明时,不会像 c 语言一样给这样const对象单独分配内存,c 语言一般是放在只读数据区,而 c ++ 编译器是把const对象放在一个符号表里面(我个人觉得放在符号表里面的其中一个原因可能是想减少一些存储操作次数),至于符号表是属于内存布局(文章:你该知道你写的程序的内存布局)中的哪一块,我也不知道,写 c++ 编译器的人才知道。

const是限定一个变量不允许改变(只读),使用const在一定程度上可以提高程序的安全性和可靠性。


// 我们先来看看const的基础知识void main(){ const int a; int const b; // 和前面一个意思一样,代表常整型数 const int *c; int const *d; // 和前面一个意思一样,表示所指向的内存数据不能被修改,但是本身可以修改 int * const e; // 指针变量不能指向其他的地址,但是它所指向内存数据可以被修改 const int * const f; // 指针变量不能指向其他的地址,它所指向内存数据也不可以被修改}


我们来做一个关于const的实验:

void main(){ const int a = 10; a = 11;}// 编译报错:error: assignment of read-only variable ‘a’


从上面代码看来const好像确实是限定一个变量不允许改变(只读),定义的变量 a 貌似变成了一个常量一样,那我们接下来继续:


void main(){ // 貌似定义的 a 是一个常量 const int a = 10; // a = 11; int *p = (int *)&a; *p = 11; // 通过指针间接赋值试试看 printf("a = %d \n", a); }// 编译成功   打印结果 a = 11 


我们发现貌似定义的 a是一个常量,但是通过指针却可以间接的修改 a 的值,const不是限定变量不允许修改吗?怎么被改了?这样看来C语言中const好像确实是一个“冒牌货”。


那么同样的代码,我们看看在C++中的表现:

void main(){ // 貌似定义的 a 是一个常量 const int a = 10; // a = 11; int *p = (int *)&a; *p = 11; // 间接赋值 printf("a = %d \n", a);   system("pause");}// 打印结果 a = 10  (结果不应该是 a = 11  ?????????)// 大家可以尝试 C 和 C++ 都进行编译对比一下。


为什么 c 和 c++ 编译的结果大相径庭?好好想想,如果是你用 c++写了一个这样的程序是用在银行后台算账的,那就麻烦大了,竟然存在这样的bug?银行每天流水那么多,账要是错了,想想都害怕吧。


其实在 c++语言里面const修饰的才算是一个真正的常量,在 c 语言中 const 可以说是个“冒牌货”。为什么会这样?其实是 c++ 编译器对 const 进行了加强,当 c++ 编译器遇到常量声明时,不会像 c 语言一样给这样const对象单独分配内存,c 语言一般是放在只读数据区,而 c ++ 编译器是把const对象放在一个符号表里面(我个人觉得放在符号表里面的其中一个原因可能是想减少一些存储操作次数),至于符号表是属于内存布局(文章:你该知道你写的程序的内存布局)中的哪一块,我也不知道,写 c++ 编译器的人才知道。


在 c++ 中使用 const 对象(比如打印这个对象)的时候,就会从符号表里面把对象的值拿出来使用,比如printf("a = %d \n", a);  ,这时候就是把 a 的值10拿出来使用,但是当你对 a 取地址(&a)的时候,c++ 编译器会为这个a单独分配一个内存空间,如果定义一个指针指向这个内存空间(int *p = (int *)&a;),那么这个指针指向的是这个新分配的一个内存空间,然后通过这个指针间接修改这个值(*p = 11;),这时候修改的其实新分配的这个空间的值,不管你间接修改的这个值是11、20、30还是100,都和符号表原本的 a 的值没有任何关系,所以使用 a 的时候打印出来的结果是 a = 10,这就是符号表,是 c++ 对 c 的一些扩展,这样就会发现 c++ 编译器把 const 变成符号表这个手段确确实实把 const 修饰的变量变成了一个常量,结论就是在 c 语言里面 const 确实是一个“冒牌货”。


这时候可能还有一个疑问,这个新分配的内存到底存不存在?这个简单,我们加一句打印就行:

void main(){ // 貌似定义的 a 是一个常量 const int a = 10; // a = 11; int *p = (int *)&a; *p = 11; // 间接赋值  printf("*p = %d \n", *p); // 加上这句打印 printf(" a = %d \n", a);   system("pause");}// 打印结果:*p = 11     a = 10


从打印可以看出单独分配的这个内存空间值是11,和原来的 a 是不同的两个概念,这就是在 C++ 中 const 的符号表的实现机制。

本站声明: 本文章由作者或相关机构授权发布,目的在于传递更多信息,并不代表本站赞同其观点,本站亦不保证或承诺内容真实性等。需要转载请联系该专栏作者,如若文章内容侵犯您的权益,请及时联系本站删除。
换一批
延伸阅读

9月2日消息,不造车的华为或将催生出更大的独角兽公司,随着阿维塔和赛力斯的入局,华为引望愈发显得引人瞩目。

关键字: 阿维塔 塞力斯 华为

加利福尼亚州圣克拉拉县2024年8月30日 /美通社/ -- 数字化转型技术解决方案公司Trianz今天宣布,该公司与Amazon Web Services (AWS)签订了...

关键字: AWS AN BSP 数字化

伦敦2024年8月29日 /美通社/ -- 英国汽车技术公司SODA.Auto推出其旗舰产品SODA V,这是全球首款涵盖汽车工程师从创意到认证的所有需求的工具,可用于创建软件定义汽车。 SODA V工具的开发耗时1.5...

关键字: 汽车 人工智能 智能驱动 BSP

北京2024年8月28日 /美通社/ -- 越来越多用户希望企业业务能7×24不间断运行,同时企业却面临越来越多业务中断的风险,如企业系统复杂性的增加,频繁的功能更新和发布等。如何确保业务连续性,提升韧性,成...

关键字: 亚马逊 解密 控制平面 BSP

8月30日消息,据媒体报道,腾讯和网易近期正在缩减他们对日本游戏市场的投资。

关键字: 腾讯 编码器 CPU

8月28日消息,今天上午,2024中国国际大数据产业博览会开幕式在贵阳举行,华为董事、质量流程IT总裁陶景文发表了演讲。

关键字: 华为 12nm EDA 半导体

8月28日消息,在2024中国国际大数据产业博览会上,华为常务董事、华为云CEO张平安发表演讲称,数字世界的话语权最终是由生态的繁荣决定的。

关键字: 华为 12nm 手机 卫星通信

要点: 有效应对环境变化,经营业绩稳中有升 落实提质增效举措,毛利润率延续升势 战略布局成效显著,战新业务引领增长 以科技创新为引领,提升企业核心竞争力 坚持高质量发展策略,塑强核心竞争优势...

关键字: 通信 BSP 电信运营商 数字经济

北京2024年8月27日 /美通社/ -- 8月21日,由中央广播电视总台与中国电影电视技术学会联合牵头组建的NVI技术创新联盟在BIRTV2024超高清全产业链发展研讨会上宣布正式成立。 活动现场 NVI技术创新联...

关键字: VI 传输协议 音频 BSP

北京2024年8月27日 /美通社/ -- 在8月23日举办的2024年长三角生态绿色一体化发展示范区联合招商会上,软通动力信息技术(集团)股份有限公司(以下简称"软通动力")与长三角投资(上海)有限...

关键字: BSP 信息技术
关闭