C++的顶层const和底层const的理解

最近,又一次翻开C++primer,决定仔细研究一下自己以前没搞懂的顶层const和底层const,这次看了后感觉明白了,所以记录下来,以后可以没事翻阅,增加记忆。首先,const是一个限定符,被它修饰的变量的值不能改变。对于一般的变量来说,其实没有顶层const和底层const的区别,而只有向指针这类复合类型的基本变量,才有这样的区别。

 一 如何区分顶层const和底层const

指针如果添加const修饰符时有两种情况:
指向常量的指针:代表 不能改变其指向内容的指针。声明时const可以放在类型名前后都可,拿int类型来说,声明时:const int和int const 是等价的。声明指向常量的指针也就是 底层const,下面举一个例子:
int num_a = 1;
int const  *p_a = &num_a; //底层const
//*p_a = 2;  //错误,指向“常量”的指针不能改变所指的对象

注意:指向“常量”的指针不代表它所指向的内容一定是常量,只是代表不能通过解引用符(操作符*)来改变它所指向的内容。上例中指针p_a指向的内容就不是常量,可以通过赋值语句:num_a=2;  来改变它所指向的内容。

2 指针常量:代表指针本身是常量,声明时必须初始化,之后它存储的地址值就不能再改变。声明时const必须放在指针符号*后面,即:*const 。声明常量指针就是顶层const,下面举一个例子:

int num_b = 2;
int *const p_b = &num_b; //顶层const
//p_b = &num_a;  //错误,常量指针不能改变存储的地址值

其实顶层const和底层const很简单, 一个指针本身添加const限定符就是顶层const,而指针所指的对象添加const限定符就是底层const。 

二 区分顶层const和底层const的作用

为啥非要区分顶层const和底层const呢,根据C++primer的解释,区分后有两个作用。
1 执行对象拷贝时有限制,常量的底层const不能赋值给非常量的底层const。也就是说,你只要能正确区分顶层const和底层const,你就能避免这样的赋值错误。下面举一个例子:
int num_c = 3;
const int *p_c = &num_c;  //p_c为底层const的指针
//int *p_d = p_c;  //错误,不能将底层const指针赋值给非底层const指针
const int *p_d = p_c; //正确,可以将底层const指针复制给底层const指针
2 使用命名的强制类型转换函数const_cast时,需要能够分辨底层const和顶层const,因为const_cast只能改变运算对象的底层const。下面举一个例子:
int num_e = 4;
const int *p_e = &num_e;
//*p_e = 5;  //错误,不能改变底层const指针指向的内容
int *p_f = const_cast<int *>(p_e);  //正确,const_cast可以改变运算对象的底层const。但是使用时一定要知道num_e不是const的类型。
*p_f = 5;  //正确,非顶层const指针可以改变指向的内容
cout << num_e;  //输出5

三 练习

说了这么多,应该练习一下,const int *const*const* pppi 是顶层const还是底层const?
答案当然是底层const,因为int前面const限定符,而最后一个*后面没有const限定符。看最后一个例子:
const int a = 1;  
//int * pi = &a;  //错误,&a是底层const,不能赋值给非底层const 
const int * pi = &a; //正确,&a是底层const,可以赋值给底层const
const int *const *const ppi = &pi  //即是底层const,也是顶层const
const int  *const *const *pppi = &ppi; //底层const
说了这么多,大家清楚了吧?

  • 129
    点赞
  • 291
    收藏
    觉得还不错? 一键收藏
  • 36
    评论
### 回答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是程序员必备的功夫之一。
评论 36
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值