C++顶层const和底层const

《C++primer》中写到:顶层const表示指针本身是个常量,底层const表示指针所指的对象是一个常量。
顶层const可以表示任意的对象是常量,这一点对任何数据类型都适用。底层const则与指针和引用等复合类型的基本类型部分有关,比较特殊的是,指针类型既可以是顶层const也可以是底层const

int i = 0;
int *const p1 = &i; // 顶层const
const int ci = 42; // 顶层const
const int *p2 = &ci; // 底层const
const int *const p3 = p2; // 靠右的const是顶层const,靠左的const是底层const
const int &r = ci;        // 用于声明引用的const都是底层const

感觉还是好难理解,那么,如何分辨顶层const和底层const呢?
顶层const是施加在类型本身上的const。

const int a;  // const施加在类型本身int上,所以是顶层const
const int *p;  // const施加在int上,而没有施加在类型本身int *上,所以不是顶层const
int *const p;  // const施加在类型本身int *上,所以是顶层const
const int &ref;  // const施加在int上,而没有施加在类型本身int &上(当然也不可能),所以不是顶层const
const int arr[5];  // const施加在数组的元素类型int上,而没有施加在数组类型int[5]本身,所以不是顶层const
// 这里补充一句,如果对数组类型使用cv修饰符,则cv修饰符修饰的是数组的元素类型(C++20标准9.3.4.5-5)

“以*分界,把一个声明从右向左读”
注意语法,* 读作 pointer to (指向…的指针),const (常量) 是形容词,char (变量类型) 和 p (变量名) ,只要这样读,哪怕是初中生也分得清了

  1. const char * p 读作:p is a pointer to a const char,译:p 是一个指针(变量),它指向一个常量字符(const char)
  2. char * const p 读作:p is a const pointer to a char,译:p 是一个常量指针(const p),它指向一个字符(变量)。
const int v2 = 0; // 顶层const
const int *p2 = &v2, *const p3 = &i, &r2 = v2;
// 分解一下
const int *p2 = &v2; // p2指向的内容不可变,顶层const
const int * const p3 = &i; // p3指向的内容和p3本身都不可变,既是顶层const也是底层const
const int &r2 = v2; // r2是一个常引用,它的值不可变,是一个顶层const
// int v1 = v2;
// int *p1 = &v1;
// const int *p2 = &v2;
p1 = p2;  // 非法
          // p2是一个底层const,它指向的内容不可变,强约束
          // p1是个普通整型指针,属松散约束
          // 从强约束到宽约束的转换是不允许的
// 编译器报错
// error C2440: '=': cannot convert from 'const int *' to 'int *'
p1 = p3; // 用p1接收p2都已经非法了,何况p3既是底层const也是顶层const
p2 = p1; // 合法,p2是底层const,指针p2本身是可以变化的
p2 = p3; // 合法
         // p3是双const约束,内容和指针本身都不可以变
         // p2是底层const,它控制的是指向的内容的只读性
         // 用p2接收p3, 各司其职,非常丝滑

为什么要区分顶层const和底层const呢,有两个作用:

  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指针
  1. 使用命名的强制类型转换函数const_cast时,需要能够分辨底层const和顶层const,因为const_cast只能改变运算对象的底层cosnt
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
  • 9
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 打赏
    打赏
  • 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
发出的红包

打赏作者

Qi妙代码

你的鼓励将是我创作的最大动力

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

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

打赏作者

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

抵扣说明:

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

余额充值