2.4 const限定符

2.4 const限定符

有时我们希望定义这样一种变量,它的值不能被改变。例如,用一个变量来表示缓冲区的大小。使用变量的好处是当我们觉得缓冲区大小不再合适时,很容易对其进行调整。另一方面,也应随时警惕防止程序一不小心改变了这个值。为了满足这一要求,可以用关键词const对变量的类型加以限定:

	const int bufSize = 512; // 输入缓冲区大小

这样就把bufSize定义成了一个常量。任何试图为bufSize赋值的行为都将引发错误:

	bufSize = 512; // 错误: 试图向const变量赋值

因为const对象一旦创建后其值就不能再改变,所以const对象必须初始化。一如既往,初始值可以是任意复杂的表达式:

	const int i = get_size(); // 正确: 运行时初始化
	const int j = 42; // 正确: 编译时初始化
	const int k; // 错误: k是一个未经初始化的常量

初始化和const

正如之前反复提到的,对象的类型决定了它支持的操作。与非const类型所能参与的操作相比,const类型的对象能完成其中大部分,但也不是所有的操作都适合。主要的限制就是只能在const类型的对象上执行不改变其内容的操作。例如,const int和普通的int一样都能参与算术运算,也都能转换成一个布尔值,等等。
在不改变const对象的操作中还有一种是初始化,如果利用一个对象去初始化另外一个对象,则它们是不是const都无关紧要:

	int i = 42;
	const int ci = i; // 正确: i的值被拷贝给了ci
	int j = ci; // 正确: ci的值被拷贝给了j

尽管ci是整型变量,但无论如何ci中的值还是一个整型数。ci的常量特征仅仅在执行改变ci的操作时才会发挥作用。当用ci去初始化j时,根本无需在意ci是不是一个常量。拷贝一个对象的值并不会改变它,一旦拷贝完成,新的对象就和原来的对象没什么关系了。

默认状态下,const对象仅在文件内有效

当以编译时初始化的方式定义一个const对象时,就如对bufSize的定义一样:

	const int bufSize = 512; // 输入缓冲区大小

编译器在编译过程中把用到该变量的地方都替换成对应的值。也就是说,编译器会找到代码中所有用到bufSize的地方,然后用512替换。
为了执行上述替换,编译器必须知道变量的初始值。如果程序包含多个文件,则每个用了const对象的文件都必须得访问到它的初始值才行。要做到这一点,就必须在每一个用到变量的文件中都有对它的定义(参见2.2.2节)。为了支持这一用法,同时避免对同一变量的重复定义,默认情况下,const对象被设定为仅在文件内有效。当多个文件中出现了同名的const变量时,其实等同于在不同文件中分别定义了独立的变量。
某些时候有这样一种const变量,它的初始值并不是一个常量表达式,但又确实有必要在文件间共享。在这种情况下,我们不希望编译器为每个文件分别生成独立的变量。相反,我们想让这类const对象像其他(非常量)对象一样工作,也就是说,只在一个文件定义const,而在其他多个文件中声明并使用它
解决的办法是,对于const变量不管是声明还是定义都添加extern关键字,这样只需定义一次就可以了:

	// 某个文件定义并初始化了一个常量,该常量能被其他文件访问
	extern const int bufSize = fun();
	// 另外一个文件
	extern const int bufSize; // 与上面定义的bufSize是同一个

如上述程序所示,第一个文件定义并初始化了bufSize。因为这条语句包含了初始值,所以它(显然)是一次定义。然而,因为bufSize是一个常量,必须用extern加以限定使其能被其他文件使用。
后面的文件中的声明也由extern做了限定,其作用是指明bufSize并非本文件所独有,它的定义将在别处出现。

总结:如果想在多个文件之间共享const对象,必须在变量的定义之前添加extern关键字。

2.4节练习


练习2.26:下面哪些语句是合法的?如果有不合法的句子,请说明理由。

	(a) const int buf;			(b) int cnt = 0;
	(c) const int sz = cnt;		(d) ++cnt; ++sz;

2.4.1 const的引用

可以把引用绑定到const对象上,就像绑定到其他对象上一样,我们称之为对常量的引用reference to const)。与普通引用不同的是,对常量的引用不能用作修改它所绑定的对象:

	const int ci = 1024;
	const int &r1 = ci; // 正确: 引用及其对应的对象都是常量
	r1 = 42; // 错误: r1是对常量的引用
	int &r2 = ci; // 错误: 试图让一个非常量引用指向一个常量对象

因为不允许直接为ci赋值,当然也就不能通过引用去改变ci。因此,对r2的初始化是错误的。假设该初始化合法,则可以通过r2来改变它引用对象的值,这显然是不正确的。

术语:常量引用是对const的引用


C++程序员们经常把词组“对const的引用”简称为“常量引用”,这一简称还是挺靠谱的,不过前提是得时刻记得这就是个简称而已。
严格来说,并不存在常量引用。因为引用本身不是一个对象,所以我们没法让引用本身恒定不变。事实上,由于C++语言并不允许随意改变引用所绑定的对象,所以从这层意义上理解所有的引用又都算是常量。引用的对象是常量还是非常量可以决定其所能参与的操作,却无论如何都不会影响到引用和对象的绑定关系本身

初始化和对const的引用

2.3.1节提到,引用的类型必须与其所引用对象的类型一致,但是有两个例外。第一种例外情况就是在初始化常量引用时允许用任意表达式作为初始值,只要该表达式能转换成引用的类型即可。尤其,允许为一个常量引用绑定非常量的对象、字面值,甚至是一个表达式:

	int i = 42;
	const int &r1 = i; // 允许将const int&绑定到一个普通int对象上
	const int &r2 = 42; // 正确: r1是一个常量引用
	const int &r3 = r1 * 2; // 正确: r3是一个常量引用
	int &r4 = r1 * 2; // 错误: r4是一个普通的非常量引用

要想理解这种例外情况的原因,最简单的方法是弄清楚当一个常量引用被绑定到另外一种类型上时到底发生了什么

	double dval = 3.14;
	const int &ri = dval;

此处ri引用了一个int型的整数。对ri的操作应该是整数运算,但dval是一个双精度浮点数而非整数。因此为了确保让ri绑定一个整数,编译器把上述代码却变成了如下形式:

	double dval = 3.14;
	const int temp = dval; // 由双精度浮点数生成一个临时的整型变量
	const int &ri = temp; // 让ri绑定这个临时量

在这种情况下,ri绑定了一个临时量temporary)对象。所谓临时量对象就是当编译器需要一个空间来暂存表达式的求值结果时临时创建的一个未命名的对象。C++程序员们常把临时量对象简称为临时量。
接下来探讨当ri不是常量时,如果执行了类似于上面的初始化过程将带来什么样地后果。如果ri不是常量,就允许对ri赋值,这样就会改变ri所引用对象的值。注意,此时绑定的对象是一个临时量而非dval。程序员既然让ri引用dval,就肯定想通过ri改变dval的值,否则干什么要给ri赋值呢?如此看来,既然大家基本上不会想着把引用绑定到临时量上,C++语言也就把这种行为归为非法

const的引用可能引用一个并非const的对象

必须认识到,常量引用仅对引用可参与的操作做出了限定,对于引用的对象本身是不是一个常量未作限定。因为对象也可能是个非常量,所以允许通过其他途径改变它的值:

	int i = 42;
	int &r1 = i; // 引用ri绑定对象i
	const int &r2 = i; // r2也绑定对象i, 但不允许通过r2修改i的值
	r1 = 0; // r1并非常量,i的值修改为0
	r2 = 0; // 错误: r2是一个常量引用

2.4.2 指针和const

与引用一样,也可以令指针指向常量或非常量。类似于常量引用指向常量的指针pointer to const)不能用于改变其所指对象的值。要想存放常量对象的地址,只能使用指向常量的指针:

	const double pi = 3.14; // pi是个常量, 它的值不能改变
	double *ptr = π // 错误: ptr是个普通指针
	const double *cptr = π // 正确: cptr可以指向双精度常量
	*cptr = 42; // 错误: 不能给*cptr赋值

2.3.2节提到,指针的类型必须与其所指对象的类型一致,但是有两个例外。第一种例外情况是允许令一个指向常量的指针指向一个非常量对象

	double pi = 3.14; // pi是一个双精度浮点数, 它的值可以改变
	const double *cptr = π // 正确: 但是不能通过cptr改变pi的值

和常量引用一样,指向常量的指针也没有规定其所指的对象必须是一个常量。所谓指向常量的指针仅仅要求不能通过该指针改变对象的值,而没有规定那个对象的值不能通过其他途径所改变。

试试这样想吧:所谓指向常量的指针或引用,不过是指针或引用“自以为是”罢了,它们觉得自己指向了常量,所以自觉地不去改变所指对象的值。

const指针

指针是对象而引用不是,因此就像其他类型一样,允许把指针本身定为常量常量指针const pointer)必须初始化,而且一旦初始化完成,则它的值(也就是存放在指针中的那个地址)就不能再改变了。把*放在const关键字之前用以说明指针是一个常量,这样的书写形式隐含着一层意味,即不变的是指针本身的值而非指向的那个值:

	int errNum = 0;
	int *const curErr = &errNum; // curErr将一直指向errNum
	const double pi = 3.14159;
	const double *const pip=π // pip是一个指向常量对象的常量指针

此例中,离curErr最近的符号是const,意味着curErr本身是一个常量对象,对象的类型由声明符的其他部分决定。声明符中下一个符号是*,意思是curErr是一个常量指针。最后,该声明语句的基本数据类型部分确定了常量指针指向的是一个int对象。与之相似,我们也能推断出,pip是一个常量指针,它指向的对象是一个双精度浮点型常量。
指针本身是一个常量并不意味着不能通过指针修改其所指对象的值,能否这样做完全依赖于所指对象的类型。例如,pip是一个指向常量的常量指针,则不论是pip所指的对象值还是pip自己存储的那个地址都不能改变。相反的,curErr指向的是一个一般的非常量整数,那么就完全可以用curErr去修改errNumb的值:

	*pip = 2.72; // 错误: pip是一个指向常量的指针
	if (*curErr) { // 如果curErr所指的对象(也就是errNumb)的值不为0
		errorHandler();
		*curErr = 0; // 正确: 把curErr所指的对象的值重置
	}

2.4.2节练习

练习2.27:下面的哪些初始化是合法的?说明理由。

	(a) int i = -1, &r = 0;		  (b) int *const p2 = &i2;
	(c) const int i = -1, &r = 0; (d) const int *const p3 = &i2;
	(e) const int *p1 = &i2;	  (f) const int &const r2;
	(g) const int i2 = i, &r = i;

练习2.28:说明下面的这些定义是什么意思,挑出其中不合法的。

	(a) int i, *const cp;		(b) int *p1, *const p2;
	(c) const int ic, &r = ic;	(d) const int *const p3;
	(e) const int *p;

练习2.29:假设已有练习2.28中定义的那些变量,则下面哪些语句是不合法的?说明理由。

	(a) i = ic;						(b) p1 = p3;
	(c) p1 = ⁣					(d) p3 = ⁣
	(e) p2 = p1;					(f) ic = *p3;

2.4.3 顶层const

如前所述,指针本身是一个对象,它又可以指向另外一个对象。因此,指针本身是不是常量以及指针所指的是不是一个常量就是两个互相独立的问题。用名词顶层consttop-level const)表示指针本身是个常量,而用名词底层constlow-level const)表示指针所指的对象是一个常量。
更一般的,顶层const可以表示任意的对象是常量,这一点对任何数据类型都适用,如算数类型、类、指针等。底层const则与指针和引用等符合类型的基本类型部分有关。比较特殊的是,指针类型既可以是顶层const也可以使底层const,这一点和其他类型相比区别明显:

	int i = 0;
	int *const p1 = &i; // 不能改变p1的值, 这是一个顶层const
	const int ci = 42; // 不能改变ci的值, 这是一个顶层const
	const int *p2 = &ci; // 允许改变p2的值, 这是一个底层const
	const int *const p3 = p2; // 左边是底层const, 右边是顶层const
	const int &r = ci; // 用于声明引用的const都是底层const

当执行对象的拷贝操作时,常量时顶层const还是底层const区别明显。其中,顶层const不受什么影响:

	i = ci; // 正确: 拷贝ci的值, ci是一个顶层const, 对此操作无影响
	p2 = p3; // 正确: p2和p3指向的对象类型相同, p3顶层const的部分不影响

执行拷贝操作并不会改变被拷贝对象的值。因此,拷入和拷出的对象是否是常量都没什么影响。
另一方面,底层const的限制却不能忽视。当执行对象的拷贝操作时,拷入和拷出的对象必须具有相同的底层const资格,或者两个对象的数据类型必须能够转换。一般来说,非常量可以转换成常量,反之则不行:

	int *p = p3; // 错误: p3包含底层const的定义, 而p没有
	p2 = p3; // 正确: p2和p3都是底层const
	p2 = &i; // 正确: int* 能转换成 const int*
	int &r = ci; // 错误: 普通的int&不能绑定到int常量上
	const int &r2 = i; // 正确: const int& 可以绑定到一个int常量上

p3既是顶层const也是底层const,拷贝p3时可以不在乎它是个顶层const,但是必须清楚它指向的对象得是一个常量。因此,不能用p3去初始化p,因为p指向的是一个普通的(非常量)整数。另一方面,p3的值可以赋给p2,是因为这两个指针都是底层const,尽管p3同时也是一个常量指针(顶层const),仅就这次赋值而言不会有什么影响。

2.4.3节练习


练习2.30:对于下面的这些语句,请说明对象被声明成了顶层const还是底层const

	const int v2 = 0;
	int v1 = v2;
	int *p1 = &v1, &r1 = v1;
	const int *p2 = &v2, *const p3 = &i, &r2 = v2;

练习2.31:假设已有上一个练习中所做的那些声明,则下面的哪些语句是合法的?请说明顶层const和底层const在每个例子中有何体现。

	r1 = v2;
	p1 = p2; p2 = p1;
	p1 = p3; p2 = p3;

2.4.4 constexpr和常量表达式

常量表达式const expression)是指值不会改变并且在编译过程就能得到计算结果的表达式。显然,字面值属于常量表达式,用常量表达式初始化的 const对象也是常量表达式。后面将会提到,C++语言中有几种情况下是要用到常量表达式的。
一个对象(或表达式)是不是常量表达式由它的数据类型和初始值共同决定,例如:

	const int max_files = 20; // max_files是常量表达式
	const int limit = max_files + 1; // limit是常量表达式
	int staff_size = 27; // staff_size不是常量表达式
	const int sz = get_size(); // sz不是常量表达式

尽管staff_size的初始值是个字面值常量,但由于它的数据类型只是一个普通int而非const int,所以它不属于常量表达式。另一方面,尽管sz本身是一个常量,但它的具体值直到运行时才能获取到,所以也不是常量表达式。

constexpr变量

在一个复杂系统中,很难(几乎肯定不能)分辨一个初始值到底是不是常量表达式。当然可以定义一个 const变量并把它的初始值设为我们认为的某个常量表达式,但在实际应用时,尽管要求如此却常常发现初始值并非常量表达式的情况。可以这么说,在此种情况下,对象的定义和使用根本就是两回事儿。
C++11新标准规定,允许将变量声明为constexpr类型以便由编译器来验证变量的值是否是一个常量表达式。声明为 constexpr的变量一定是一个常量,而且必须用常量表达式初始化:

	constexpr int mf = 20; // 20是常量表达式
	constexpr int limit = mf + 1; // mf + 1是常量表达式
	constexpr int sz = size(); // 只有当size是constexpr函数时
							   // 才是一条正确的声明语句

尽管不能使用普通的函数作为constexpr变量的初始值,但新标准允许定义一种特殊的constexpr函数。这种函数应该足够简单以使得编译时就可以计算其结果,这样就能用constexpr函数去初始化constexpr变量了。

一般来说,如果你认定变量时一个常量表达式,就把它声明成constexpr变量。

字面值类型

常量表达式的值需要在编译时就得到计算,因此对声明constexpr时用到的类型必须有所限制。因为这些类型一般比较简单,值也显而易见、容易得到,就把它们称为“字面值类型”(literal type)。
到目前为止接触过的数据类型中,算数类型、引用和指针都属于字面值类型。自定义类Sales_item、IO库、string类型则不属于字面值类型,也就不能被定义成constexpr
尽管指针和引用都能定义成constexpr,但它们的初始值却受到严格限制。一个constexpr指针的初始值必须是nullptr或者0,或者是存储于某个固定地址中的对象。
后面将会提到,函数体内定义的变量一般来说并非存放在固定地址中,因此constexpr指针不能指向这样的变量。相反的,**定义与所有函数体之外的对象其地址固定不变,能用来初始化constexpr指针。而且,允许函数定义一类有效范围超出函数本身的变量,这类变量和定义在函数体之外的变量一样也有固定地址。因此,constexpr引用能绑定到这样的变量上,constexpr指针也能指向这样的变量。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

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

余额充值