指针和const

常量指针 & 指针常量

常量指针

常量指针 ,即指向常量的指针(make a pointer point to a contant object),即pointer-to-const

指针指向的值是常量,因此指针指向的值不可以通过指针修改,如(*pa)++*pa = &bcin >> *pa都是不允许的。但其指向的值可以通过其他方式修改,比如直接修改a的值,或者修改指针的指向pa = &b;

例子:

C++ 复制代码
int a = 12;
int b = 23;
const int* pa = &a;  // 指向a的常量指针

cout << "before, *pa = " << *pa << ", pa = " << pa << endl;  // before, *pa = 12, pa = 012FFE88
// 7-10行每次只运行其中一行
(*pa)++;  // 编译错误,不能给常量赋值
*pa++;	  // after, *pa = -858993460, pa = 012FFE92,这是因为操作符优先级,这种写法等同于*(pa++),为了避免语义不清或错误不建议这样写
a = 11;   // after, *pa = 11, pa = 012FFE88
pa = &b;  // after, *pa = 23, pa = 00F5FCB4
cout << "after, *pa = " << *pa << ", pa = " << pa << endl;

指针常量(const pointer)

指针常量(const pointer),指针本身是常量,即指针的指向不可被修改。

其实很容易理解,指针中保存的是地址,该指针为常量则该指针的内容不可被修改,即这个地址是不能被修改的,但这个地址中的内容是可以被修改的。

例子:

C++ 复制代码
int a = 12;
int b = 23;
int* const pb = &b;  // 指针本身是常量
cout << "before, *pb = " << *pb << ", pb = " << pb << endl;  // before, *pb = 23, pb = 00F5FCB4
pb = &a;  // 编译错误,不能给常量赋值
*pb = a;  // after, *pb = 12, pb = 00F5FCB4
cout << "after, *pb = " << *pb << ", pb = " << pb << endl;

指向常量的指针常量

C++ 复制代码
const float g_earth = 9.80;
const float* const pe = &g_earth;  // g_earth不可修改,也无法通过pe修改g_earth的值,pe也不能指向其他地址

这里的pe*pe都是const。

普通指针可以指向常量吗?

不行!

例子:

C++ 复制代码
const float g_earth = 9.80;
float* p = &g_earth;		 // 编译错误

假设C++允许这种方式,那么就可以通过*p来修改常量g_earth的值,C++不允许这种行为,因此指向常量的指针必须为常量指针。

但我就想让p指向g_earth,真的就没办法了吗?

可以用<const_cast>

C++ 复制代码
const float g_earth = 9.80;
float* p = const_cast<float*>(&g_earth);

常量指针可以指向非常量吗?

只有指向的非常量不是指针才可以。

例子:

C++ 复制代码
const int** pp2;
int* p1;
const int n = 13;
pp2 = &p1;	// 编译错误
*pp2 = &n;
*p1 = 10;

假设pp2 = &p1是被允许的,那么*pp2 = &n则会将p1指向n,那么就可以通过p1来修改const int n的值。

小结

常量指针(pointer-to-const)可以指向常量(const data)或非常量(non-const data),但被指向的数据不能是指针;

而普通指针(non-const pointer)只能指向非常量(non-const data)。

C++ 复制代码
int a = 3;
const int* p1 = &a;  // a pointer to const int
int* const p2 = &a;  // a const pointer to int
相关推荐
C++ 老炮儿的技术栈7 分钟前
Ubuntu root账号自动登陆
linux·运维·服务器·c语言·c++·ubuntu·visual studio
Irissgwe37 分钟前
map/set/multimap/multiset 的底层逻辑与实现
数据结构·c++·算法·二叉树·stl·c·红黑树
凡人叶枫1 小时前
Effective C++ 条款39:明智而审慎地使用 private 继承
java·数据库·c++·嵌入式开发
不想写代码的星星1 小时前
伪共享:逻辑无共享,物理打成狗
c++
noipp2 小时前
【无标题】
c语言·数据结构·c++·算法
森G2 小时前
64、完善聊天室程序(TLV拓展)---------网络编程
网络·c++·tcp/ip
郝学胜-神的一滴3 小时前
完全二叉树与堆底层原理深度剖析 | 手写C++大顶堆实现
java·开发语言·数据结构·c++·python·算法
大白话_NOI3 小时前
【洛谷 P2678】 [NOIP2015 提高组] 跳石头 超详细题解
c++·算法
chase_my_dream4 小时前
LeGO-LOAM 详细源码流程解读
c++·计算机视觉·自动驾驶
插件开发4 小时前
vs2015 cuda c++ 线程号的计算详解
开发语言·c++·算法