C和C++中const的实现机制

本文探讨了C和C++中const关键字的实现机制及其差异。在C中,const主要在编译阶段起作用,防止直接修改变量,但实际上并不阻止间接修改。而在C++中,const变量会被视为真正的常量,编译器尝试进行常量折叠,对于内置类型,const值在编译时替换,但对于自定义类型则在运行时分配内存,确保不可修改。这种处理方式提高了代码效率和安全性。
摘要由CSDN通过智能技术生成

目录

C中的const实现机制

C++中的const实现机制


C中的const实现机制

        C中const修饰的变量的值不能被修改,这是const修饰的变量与普通变量的区别。但在反汇编后,发现const修饰的变量与普通变量没有区别。因此可以认为,在代码上,const修饰的变量与普通变量没有区别。那么,C中const是如何实现的呢?

有如下代码:

#include <stdio.h>
int main()
{
	const int a = 1;
	int * b = (int*)&a;
	*b = 2;
	printf("%d %d", a , *b);
	return 0;
}

输出结果: 

2 2

        上面的代码是能够通过编译并运行的,但是在输出结果中const修饰的变量a的值却变为2。但是如果直接修改变量a的值,编译器就会报错。仔细观察上面的代码,在代码中并没有直接去修改变量a的值,而是通过一个指针b间接地修改了变量a的值

        最初的C中并没有const,因为C++而在后来加入了const。实际上C并没有对const修饰的变量做过多的处理,只是在编译过程中编译器会检查是否有修改const修饰的变量的代码,如果有则报错。由此可以看出,const只是编译层面的限制,在实际编译后,const修饰的变量与普通变量没有区别。在C中,const修饰的变量是常变量,不算真正意义上的常量,不可以直接修改,但可以间接修改。

C++中的const实现机制

        在C++中,const修饰的变量的值也不能被修改,那么在C++中,const的实现机制与C中的相同吗? 

有如下代码:

#include<iostream>
using namespace std;
int main()
{
	const int a = 1;
	int * b = (int *)&a;
	*b = 2;
	cout<<a<<' '<<*b;
	return 0;
}

输出结果: 

1 2

        经过编译运行后发现变量a的值没有改变。事实上,C++是把const修饰的变量当作常量来处理的。在上面的例子中,C++先把变量a的值放在符号表里面,此时没有为变量a分配内存空间,当需要变量a时就直接从符号表里取。在编译过程中,C++将const修饰的变量替换为符号表里的值,类似于C中的#define。因此,通过指针访问的变量a的值虽然改变了,但是输出的并不是变量a的值,而是变量a经过替换后的值,也就是符号表里的值。当用指针去访问变量a时,C++就会为变量a分配内存空间,也就是所谓的常量折叠。此时指针指向的是变量a的地址,输出的值是指针所指向的地址中存放的值,即修改后的2。但是,C++能对所有的常量进行替换吗?

有如下代码:

#include <iostream>
using namespace std;

struct t
{
	int i;
};

int main()
{
	const struct t a = {1};
	int * p = (int *)(&a.i);
	*p = 2;
	cout<<a.i<<' '<<*p;
	return 0;
}

输出结果:

2 2

        编译后运行发现结构体的成员i发生了改变。由此可以得出,C++中的const只能对内置数据类型做常量替换,而对于像结构体这类自定义数据类型则无法进行常量替换。

        那么,C++为什么要这样处理const呢?在通常情况下,const将变量的值保存到编译器的符号表中,并不为变量分配内存空间。在这样的处理下,变量就成为了一个编译期间的常量,没有了对变量的存储和读取内存的操作,使得代码执行的效率得到提高。同时,变量的值被保存到符号表中,在需要变量的地方用符号表中保存的值直接替换,从而保证了变量的值不被修改,提高了代码的安全性。

        在旧版本的C中,如果想要建立一个常量,就必须利用#define预处理指令,但是编译器只做简单的替换,并不会对其进行检查。为了消除安全隐患,const便随之推出。在编译过程中,编译器会对const进行类型检查,从而减少了错误的发生。

  • 1
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

#include <bug>

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值