【重学C++】【指针】轻松理解常量指针和指针常量

简介: 【重学C++】【指针】轻松理解常量指针和指针常量

大家好,我是 同学小张,持续学习C++进阶知识和AI大模型应用实战案例,持续分享,欢迎大家点赞+关注,共同学习和进步。

重学C++系列文章,在会用的基础上深入探讨底层原理和实现,适合有一定C++基础,想在C++方向上持续学习和进阶的同学。争取让你每天用5-10分钟,了解一些以前没有注意到的细节。


前有数组指针和指针数组,现有常量指针和指针常量,就问你晕不晕?下面继续来讲解这些非常容易混淆和引起混乱的概念原理与用法。本文的内容是 常量指针和指针常量。

1. 认识常量指针和指针常量

1.1 字面理解

中文博大精深,还是在这个概念中间加个"的",就很容易理解:

  • 常量指针:常量的指针,指向常量的指针,所以指针指向的值是不变的。
  • 指针常量:指针的常量,指针是常量,指针本身是不变的。

1.2 从表达式区分

一般根据const位置的不同,可以有以下几种写法:

const int* p; //1
int const* p; //2
int * const p; //3
const int * const p; //4
int const * const p; //5

如何区分哪个是指针常量,哪个是常量指针呢?书中有个简单的方法:从右往左读,遇到p就读作 "p is a ",遇到*就读作"point to "。来实践一下:

第1个:p is a point to int const,int是常量,所以是常量指针

第2个:p is a point to cost int,int是常量,所以是常量指针

第3个:p is a const point to int,point是常量,所以是指针常量

第4个:p is a const point to int const,int和point都是常量

第5个:p is a const point to const int,int和point都是常量

其实还有更简单的方法:看const是在*的右边,就是修饰*,就是指针是不可变的,也就是指针常量

2. 常量指针

常量指针,常量的指针,本质是一个指针,指向一个常量,也就是指针指向的值不能变。以下是几个细节点:

(1)指针指向的值不能改变

所以,以下代码错误:

int a = 10;
const int* ptr = &a;
*ptr = 20; // error,常量指针不能修改指向的常量的值

(2)虽然指针指向的值不能变,但是指针本身的值(指向的地址)可以变

(3)如果要指向常量,必须使用常量指针

const int b = 10;
//    int *p = &b; // error
const int *p = &b; // 必须用常量指针

3. 指针常量

指针常量,这个指针是一个常量,也就是指针本身的值不能变,指针指向的值可以变。

(1)指针指向的值可以变,以下代码OK

int *const const_p = &a;
*const_p = 20;

(2)指针自身的值不能变,以下代码error

int c = 30;
const_p = &c;

(3)不能指向常量,以下代码error

const int b = 10;
int *const const_p2 = &b;

4. 总结

本文我们梳理了常量指针与指针常量,从如何区分它们到使用中的一些细节。再总结一下两者的区别:

指针常量,指针是一个常量,本身的值(指向的地址)不能变,指向的值可以变。

常量指针,指向常量的指针,本身的值(指向的地址)可以变,指向的值不能变。

其实,再简单一点,我们都知道指针一般有4种属性:自身的地址,自身的类型,自身的值(指向的地址),指向的值。这两种指针类型,我们只需关注指向的地址指向的值是否能修改即可。

5. 补充知识

5.1 疑问解答

上篇文章(【重学C++】【指针】详解让人迷茫的指针数组和数组指针)的最后,我留了一个问题:

a为一维数组,为什么下面的打印a&a地址是相同的?

std::printf("a的地址:%p\n", a);  // 输出:00000000005ffe40
std::printf("a的地址:%p\n", &a);  // 输出:00000000005ffe40

现在我说一下我的一些理解(欢迎补充和指正):

  • 询问GPT的回复:在C++中,对于一维数组a[10]={},打印出a的地址和&a的地址是相同的,是因为数组名a在C++中会被隐式转换为指向数组首元素的指针,因此a和&a的值是相同的。这种关系只在一维数组时生效,多维数组不行。

那二者有什么不同吗?我们做如下测试:

int a[5] = {0,1,2,3,4};
std::printf("a+1的地址:%p\n", a+1); // 00000000005ffe44
std::printf("&a+1的地址:%p\n", &a+1); // 00000000005ffe54
std::printf("sizeof(a)=%llu\n", sizeof(a)); // 20
std::printf("sizeof(&a)=%llu\n", sizeof(&a)); // 8

从sizeof来看,a代表的是整个数组,&a代表的是指针自身。

从+1来看,a+1是往后移一个元素的大小,&a+1是往后移整个数组的大小,所以,是否可以认为:a代表的是数组首元素的地址,而&a代表的是整个数组的首地址?

5.2 数组名的本质:到底是不是指针?

看到网上很多关于数组名本质的讨论,主要集中在数组名到底是不是一个真正的指针?

一些说法是数组名的本质是一个指针常量,因为它代表的是数组首元素的地址,且不可++操作,并且它的使用方法与指针基本无异。

另一种说法是数组名不是指针,只是代表了数组首元素地址而已,因为其sizeof的大小不是指针的大小,而是数组的大小。

针对这个争论,我认为这篇文章解释的比较清楚,可以看看:https://www.51cto.com/article/277404.html。不过我觉得,无论它是不是指针,只要不耽误会用就够了…

提醒一句:一定要动手去实践一下!没有任何一篇文章看了之后就能彻底搞懂指针,必须亲身体验才能加深印象!

如果觉得本文对你有帮助,麻烦点个赞和关注呗 ~~~


  • 大家好,我是 同学小张,持续学习C++进阶知识AI大模型应用实战案例
  • 欢迎 点赞 + 关注 👏,持续学习持续干货输出
  • 一起交流💬,一起进步💪。
  • 微信公众号也可搜【同学小张】 🙏

本站文章一览:

相关文章
|
20天前
|
安全 程序员 C语言
【C语言】指针的爱恨纠葛:常量指针vs指向常量的指针
在C语言中,“常量指针”和“指向常量的指针”是两个重要的指针概念。它们在控制指针的行为和数据的可修改性方面发挥着关键作用。理解这两个概念有助于编写更安全、有效的代码。本文将深入探讨这两个概念,包括定义、语法、实际应用、复杂示例、最佳实践以及常见问题。
40 7
|
5天前
|
存储 程序员 C++
深入解析C++中的函数指针与`typedef`的妙用
本文深入解析了C++中的函数指针及其与`typedef`的结合使用。通过图示和代码示例,详细介绍了函数指针的基本概念、声明和使用方法,并展示了如何利用`typedef`简化复杂的函数指针声明,提升代码的可读性和可维护性。
23 0
|
1月前
|
存储 编译器 Linux
【c++】类和对象(上)(类的定义格式、访问限定符、类域、类的实例化、对象的内存大小、this指针)
本文介绍了C++中的类和对象,包括类的概念、定义格式、访问限定符、类域、对象的创建及内存大小、以及this指针。通过示例代码详细解释了类的定义、成员函数和成员变量的作用,以及如何使用访问限定符控制成员的访问权限。此外,还讨论了对象的内存分配规则和this指针的使用场景,帮助读者深入理解面向对象编程的核心概念。
88 4
|
2月前
|
存储 安全 编译器
在 C++中,引用和指针的区别
在C++中,引用和指针都是用于间接访问对象的工具,但它们有显著区别。引用是对象的别名,必须在定义时初始化且不可重新绑定;指针是一个变量,可以指向不同对象,也可为空。引用更安全,指针更灵活。
|
2月前
|
存储 C++
c++的指针完整教程
本文提供了一个全面的C++指针教程,包括指针的声明与初始化、访问指针指向的值、指针运算、指针与函数的关系、动态内存分配,以及不同类型指针(如一级指针、二级指针、整型指针、字符指针、数组指针、函数指针、成员指针、void指针)的介绍,还提到了不同位数机器上指针大小的差异。
61 1
|
2月前
|
存储 编译器 C语言
C++入门2——类与对象1(类的定义和this指针)
C++入门2——类与对象1(类的定义和this指针)
44 2
|
2月前
|
存储 安全 编译器
【C++】C++特性揭秘:引用与内联函数 | auto关键字与for循环 | 指针空值(一)
【C++】C++特性揭秘:引用与内联函数 | auto关键字与for循环 | 指针空值
|
2月前
|
存储 C++ 索引
C++函数指针详解
【10月更文挑战第3天】本文介绍了C++中的函数指针概念、定义与应用。函数指针是一种指向函数的特殊指针,其类型取决于函数的返回值与参数类型。定义函数指针需指定返回类型和参数列表,如 `int (*funcPtr)(int, int);`。通过赋值函数名给指针,即可调用该函数,支持两种调用格式:`(*funcPtr)(参数)` 和 `funcPtr(参数)`。函数指针还可作为参数传递给其他函数,增强程序灵活性。此外,也可创建函数指针数组,存储多个函数指针。
|
3月前
|
编译器 C++
【C++核心】指针和引用案例详解
这篇文章详细讲解了C++中指针和引用的概念、使用场景和操作技巧,包括指针的定义、指针与数组、指针与函数的关系,以及引用的基本使用、注意事项和作为函数参数和返回值的用法。
54 3
|
2月前
|
存储 编译器 程序员
【C++】C++特性揭秘:引用与内联函数 | auto关键字与for循环 | 指针空值(二)
【C++】C++特性揭秘:引用与内联函数 | auto关键字与for循环 | 指针空值