C++的const关键字允许指定一个语义约束,编译器会强制实施这个约束,允许程序员告诉编译器某值是保持不变的。如果在编程中确实有某个值保持不变,就应该明确使用const,这样可以获得编译器的帮助。
对于基础数据类型的 const:
const修饰指针变量有以下三种情况:
修饰形参时,用相应的变量初始化const常量,在函数体内,按照const所修饰的部分进行常量化。
修饰函数返回值时,一般情况下,const修饰返回值多用于操作符的重载。
const修饰类对象时,其对象中的任何成员都不能被修改,对于对象指针和对象引用也是一样。const修饰的对象,该对象的任何非const成员函数都不能调用该对象,因为任何非const成员函数都会有修改成员变量的可能。
const修饰的成员变量不能被修改,同时只能在初始化列表中被初始化,因为常量只能被初始化,不能被赋值。
赋值是使用新值覆盖旧值,构造函数先为成员变量开辟空间然后为其赋值,不是初始化。而初始化列表开辟空间和初始化是同时完成的,直接给予一个值,所以const成员变量一定要在初始化列表中完成。
const成员函数表示该成员函数不能修改类对象中的任何非const成员变量。一般const写在函数的后面:
如果某个成员函数不会修改成员变量,那么最好将其声明为const,因为const成员函数不会对数据进行修改,如果修改,编译器将会报错。
对于const类对象,只能调用类中的const成员函数,所以const修饰成员函数的作用主要就是限制对const对象的使用。
define宏是在预处理阶段展开,define常量从未被编译器看见,因为在预处理阶段就已经被替换了,而const常量在编译阶段使用。
define宏没有类型,不做类型安全检查,仅仅是字符替换,并且在字符替换时可能会产生意料不到的错误。const常量有明确的类型,在编译阶段会进行类型检查。
define宏是字符替换,有多少地方使用,就会替换多少次,不会分配内存。
编译器通常不会为const常量分配空间,只是将它们保存在符号表内,使他们成为一个编译期间的一个常量,没有读取内存的操作,效率高。
/
*
修改局部空间的const常量,赋值时生效,使用时依然使用符号表中的常量,输出结果不变。
*
/
const
int
nVal
=
9
;
*
(
int
*
)&nVal
=
8
;
cout <<
"nVal="
<< nVal << endl;
/
*
修改局部空间的const常量,赋值时生效,使用时依然使用符号表中的常量,输出结果不变。
*
/
const
int
nVal
=
9
;
*
(
int
*
)&nVal
=
8
;
cout <<
"nVal="
<< nVal << endl;
int
nValA
=
8
;
int
nValB
=
9
;
const
int
*
pConstVal
=
&nValA;
pConstVal
=
&nValB;
/
/
*
pConstVal
=
nValB 不行
int
*
const constpVal
=
&nValA;
*
constpVal
=
nValB;
/
/
constpVal
=
&nValB 不行
const
int
*
const constpConstVal
=
&nValA;
/
/
*
constpConstVal
=
nValB 不行
/
/
constpConstVal
=
&nValB 不行
int
nValA
=
8
;
int
nValB
=
9
;
const
int
*
pConstVal
=
&nValA;
pConstVal
=
&nValB;
/
/
*
pConstVal
=
nValB 不行
int
*
const constpVal
=
&nValA;
*
constpVal
=
nValB;
/
/
constpVal
=
&nValB 不行
const
int
*
const constpConstVal
=
&nValA;
/
/
*
constpConstVal
=
nValB 不行
[注意]APP应用上架合规检测服务,协助应用顺利上架!
最后于 2021-8-24 11:30
被流照君编辑
,原因: 更改了关于修饰普通类型变量的部分情况