关于const的犄角旮旯

关于const的犄角旮旯

const关键字为程序的健壮性和编译优化提供辅助参考信息。

const主要由编译器来解读使用,是一个静态的概念。也就是,当我们对一个变量通过const说明为常量时,后续对该变量的修改,会被编译器检查出来,然后报错。
但是,一旦检查通过了,在实际运行时,跟普通变量并没有差别。比如,如果我们通过别的手段获取了常量的地址,还是可以在运行时改变它的值。内存中,它还是表现为一个普通的数据变量。

const的含义网上有很多说明文章。权威的解释,个人觉得应该是标准说明。编译器要根据标准要求,对代码中使用的const进行检查处理。

const可以修饰变量和函数。
修饰变量,可以是内置类型,也可以是自定义类型。
内置类型可以是普通变量,也可以是指针变量,还可以是引用等。

修饰函数,可以修饰函数的返回值,函数的参数,函数本身。
函数可以是C函数,可以是C++中的函数,C++中就包括静态函数和成员函数。

对函数的返回值和参数的修饰,个人理解是基于变量的修饰延伸而来,很多地方可以参考变量的修改来理解。
对于函数本身的修改,主要是用于C++中,其实是修饰的this指针。这样,我们理解const函数中不可修改成员变量,就简单了。否则,简单死记硬背,效果并不好。

这里主要看一下对于函数返回值的修饰。为了引出问题,先看几个简单例子:
对普通内置类型的修饰,比如
int const cint1 = 1;
const int cint2 = 2;
编译器会要求定义时有初值,且在代码的其他地方,只能引用,不能修改值,否则会出错。同之前说明,这些都是编译检查。
对于上述两种写法,编译器似乎都是作为const变量来处理,没有差别。这一点,后面会看到。这里先提一下。

但是,对于指针变量,上述两种写法就有不同了
int * const cpi1 = &xx;
const int * cpi2;
第一个需要定义时赋值,否则编译器会告诉你常量cpi1没有初始化。所以,这里cpi1是一个常量,不是变量,指向的地方一开始就要明确了。
第二个可以编译通过,说明cpi2是一个变量,不是常量。但是,如果尝试给其指向的内容赋值时,编译会报错,提示*cpi2是常量,也就是指针指向的内容是常量。
因为本文是研究const的犄角旮旯,所以我们看看const放在中间位置啥情况:
int const * cpi3;
实际上,这种写法很少见,有点不伦不类的,编译器将其按cpi2一样处理。

下面,我们看看const修饰函数的情况。
对于普通内置类型,我们可以凭直觉,写出如下例子:
const int func();
但是,跟变量类似,也可以有下面的可能:
int const func();
前面已经说了,对于函数本身的修饰,是放在函数的后面的,也就是类似这样的,int func() const,所以这里都是修饰返回值。
为了看出上面两种写法的差异,我们制造错误,让编译器告诉我们错误在哪里,通过错误信息,进一步判断差异。
如何制造错误?
有一种办法,我们声明一个函数指针,然后定义一个函数,让函数指针指向函数,如果二者定义不一致,编译就会报错,这里就是切入点。
typedef const int (*pfunc)();
const int getVal2()  {return 123;}

pfunc xxx;
xxx = getVal2;
因为上面类型一致,所以编译通过,我们将声明里面的const去掉,看看
typedef int (*pfunc)();
编译报错
invalid conversion from ‘const int (*)()’ to ‘pfunc {aka int (*)()}’
我们再将定义里的const挪个位置,看看
编译报了同样的错误
invalid conversion from ‘const int (*)()’ to ‘pfunc {aka int (*)()}’
这说明两种写法,在编译器看来没有差别。
好了,我们将基本内置类型换为指针,看看会怎么样
const int * getVal2()  {return &g_var;}
编译器的错误如下:
invalid conversion from ‘const int* (*)()’ to ‘pfunc {aka int (*)()}’
将const换个位置,
int const * getVal2()  {return &g_var;}
invalid conversion from ‘const int* (*)()’ to ‘pfunc {aka int (*)()}’
错误没有变化,这跟前面所述的指针变量情况是一致的
我们再看看,const放到*后面的情况
int * const getVal2()  {return &g_var;}
编译器错误变化了,
invalid conversion from ‘int* const (*)()’ to ‘pfunc {aka int (*)()}’
这三种(本质上是两种)写法的差异在哪里?

对于第一和第二种,也就是const在前面的情况,
int * presult = getVal2();
这种写法会报编译错误,invalid conversion from ‘const int*’ to ‘int*’
返回的是一个指针,指针指向的数据是常量,不能修改,左值也要求是一个同类型的指针。

对于第三种写法,上述左值编译不会有问题。const 修饰函数名,函数名本身是不可修改的,所以这种写法应该没有意义。
但是编译器会检查该修饰,在相关使用地方,均需要保持类型一致。


其他的用法?待补充。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

打赏作者

龙赤子

你的小小鼓励助我翻山越岭

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

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

打赏作者

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

抵扣说明:

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

余额充值