C++干货系列——一文完美解析“顶层”和“底层“ const

C++ 顶层 const 和底层 const

从 const 指针开始说起。const int* pInt;int *const pInt = &someInt;,前者是 *pInt 不能改变,而后者是 pInt 不能改变。因此指针本身是不是常量和指针所指向的对象是不是常量就是两个互相独立的问题。用顶层表示指针本身是个常量,底层表示指针所指向的对象是个常量。

更一般的,顶层 const 可以表示任意的对象是常量,这一点对任何数据类型都适用;底层 const 则与指针和引用等复合类型有关,比较特殊的是,指针类型既可以是顶层 const 也可以是底层 const 或者二者都是。

拷贝与顶层和底层 const

int i = 0;
int *const p1 = &i; 	//	不能改变 p1 的值,这是一个顶层
const int ci = 42;		//	不能改变 ci 的值,这是一个顶层
const int *p2 = &ci;	//	允许改变 p2 的值,这是一个底层
const int *const p3 = p2;	//	靠右的 const 是顶层 const,靠左的是底层 const
const int &r = ci;		//	所有的引用本身都是顶层 const,因为引用一旦初始化就不能再改为其他对象的引用,这里用于声明引用的 const 都是底层 const

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

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

与此相对的,底层 const 的限制却不能被忽视。当执行对象的拷贝操作时,拷入和拷出的对象必须具有相同的底层 const 资格,或者两个对象的数据类型必须能够转换,一般来说,非常量可以转化为常量,反之不行。

int *p = p3;	//	错误:p3 包含底层 const 的定义,而p没有。假设成功,p 就可以改变 p3 指向的对象的值。
p2 = p3;			//	正确:p2 和 p3 都是底层 const
p2 = &i;			//	正确:int* 能够转化为 const int*,这也是形参是底层const的函数形参传递外部非 const 指针的基础。
int &r = ci;	// 	错误:普通 int& 不能绑定到 int 常量中。
const int &r2 = i;	//	正确:const int& 可以绑定到一个普通 int 上。

函数重载与顶层和底层 const 形参

顶层 const 不影响传入函数的对象,一个拥有顶层 const 的形参无法和另一个没有顶层 const 的形参区分开:

Record lookup(Phone);
Record lookup(const Phone);			//重复声明了Record lookup(Phone)

Record lookup(Phone*);
Record lookup(Phone* const);		//该const是顶层,重复声明了Record lookup(Phone* const)

另一方面,如果形参是某种类型的指针或引用,则通过区分其是否指向的是常量对象还是非常量对象可以实现函数重载。此时的const是底层的。

Record lookup(Phone&);
Record lookup(const Phone&);		//正确,底层const实现了函数的重载

Record lookup(Phone*);
Record lookup(const Phone*);		//正确,底层const实现了函数的重载

函数重载与强制类型转换

以下节选自《Effective C++》,假设要重载一个函数,而重载函数的区别仅在于函数形参一个是底层const一个是非底层的,那么最佳的做法是采用代码的复用,减少这两个重载函数实现过程的相同代码段,可以通过const_cast来实现这一做法,在非const函数中调用const。

class Textbook
{
public:
  	...
  	const char& operator[](std::size_t position) const	 
    {
      	...
        return text[position];
    }
  	char& operator[](std::size_t position)
    {
      	//这里首先将op[]的返回值的const去除,因为const char&不能自动转换为char&,除此之外,由于调用以上const函数的必须是const对象,因此将调用此函数的对象(即*this)强制转化为一个(底层)const对象,再调用const op[]。
      	return
          	const_cast<char&>(static_cast<const TextBlock&>(*this)[position]);
    }
};

相关问题:

为什么不能在一个常量对象中调用非常成员函数?

因为在默认情况下,this的类型是指向类的非常量版本的常量指针(意思是this的值不能改变,永远指向那个对象,即“常量指针”,但是被this指向的对象本身是可以改变的,因为是非常量版本,这里this相当于是顶层const),而this尽管是隐式的,它仍然需要遵循初始化规则,普通成员函数的隐式参数之一是一个底层非const指针,在默认情况下我们无法把一个底层const的this指针转化为非const的this指针,因此我们不能在常量对象上调用普通的成员函数。因此在上例中,形参列表后的const就意味着默认this指针应该是一个底层const, 类型是 const ClassName&。而非常对象却可以调用常成员函数,因为底层非const可以默认转化为底层const。

  • 2
    点赞
  • 8
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: C++顶层const底层const都是指const修饰的对象或指针。 顶层const指的是const修饰的对象本身是不可修改的,但是可以修改指向该对象的指针。例如: ``` const int a = 10; // a是顶层const int* p = &a; // 错误,不能将const int*转换为int* const int* p = &a; // 正确,p是指向const int的指针,可以指向a *p = 20; // 错误,a是const int类型,不可修改 p = nullptr; // 正确,p本身不是const,可以修改 ``` 底层const指的是const修饰的指针指向的对象是不可修改的,但是指针本身可以修改。例如: ``` int a = 10; int b = 20; int* const p = &a; // p是底层const,指向a,不可修改 *p = 30; // 正确,a的值被修改为30 p = &b; // 错误,p是const指针,不可修改 ``` 需要注意的是,顶层const底层const可以同时存在,例如: ``` const int* const p = &a; // p是指向const int的const指针,不可修改指向的对象和指针本身 ``` ### 回答2: C++const关键字,可以作为类型修饰符,用来修饰声明对象是否可被修改。根据const修饰的位置,分为顶层const底层const顶层const,即const在*号左边。在指针本身不可修改时使用,表示指针本身不可修改,也就是指针所指向的地址可以修改,指向的值也可以修改。例如: ``` const int* p; ``` p是一个指向int类型的const指针,指针本身不可修改,但可以指向其他不同的int类型变量,同时该int类型变量也不可修改。 底层const,即const在*号右边。在指针指向的数据不可修改时使用,表示指针所指向的数据不可修改。例如: ``` int* const p; ``` p是一个指向int类型的const指针,指针本身可修改,但是指向的int变量不可修改。 同时,也可以将顶层const底层const一起使用,例如: ``` const int* const p; ``` p是一个指向int类型的const指针,该指针本身不可修改,同时也不能修改所指向的int变量。 总的来说,const关键字可以帮助我们保护变量的值不受非法修改,提高代码的可维护性和可读性,而顶层const底层const则分别用于指针和指针指向的数据不可修改的情况。 ### 回答3: C++const关键字有两种应用方式:顶层const底层const顶层const表示的是指针本身是一个常量,即指针所指向的对象不能被修改,但指针本身的指向可以改变。例如,对于int* const p,p是一个指向int的常量指针,表示p所指向的int类型的变量不能被修改,但是p可以指向其他int类型的变量。 底层const表示的是指针所指向的对象是个常量,即指针所指向的对象不能被修改,但是指针本身的指向可以改变。例如,对于const int* p,p是一个指向常量int的指针,表示p所指向的int类型的变量不能被修改,但是p可以指向其他int类型的变量。 由于顶层const底层const有不同的应用场景,它们的转换也有细微的差别。const int*可以隐式地转换成int*,即底层const可以被忽略,因为我们可以将一个指向常量的指针赋值给一个不是指向常量的指针,但int* const不能隐式地转换成const int*,因为顶层const不能被忽略。如果想要转换成const int*,需要使用强制类型转换。 总之,顶层const底层const针对的是指针和指针所指向的对象的常量性,它们的使用场合和转换规则也有区别。在C++熟练使用这两种const是程序员必备的功夫之一。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值