【重学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大模型应用实战案例
  • 欢迎 点赞 + 关注 👏,持续学习持续干货输出
  • 一起交流💬,一起进步💪。
  • 微信公众号也可搜【同学小张】 🙏

本站文章一览:

相关文章
|
25天前
|
存储 安全 C++
C++中的引用和指针:区别与应用
引用和指针在C++中都有其独特的优势和应用场景。引用更适合简洁、安全的代码,而指针提供了更大的灵活性和动态内存管理的能力。在实际编程中,根据需求选择适当的类型,能够编写出高效、可维护的代码。理解并正确使用这两种类型,是掌握C++编程的关键一步。
23 1
|
25天前
|
数据采集 存储 编译器
this指针如何使C++成员指针可调用
本文介绍了C++中的this指针,它是一个隐藏的指针,用于在成员函数中访问对象实例的成员。文章通过代码示例阐述了this指针的工作原理,以及如何使用指向成员变量和成员函数的指针。此外,还提供了一个多线程爬虫示例,展示this指针如何使成员指针在对象实例上调用,同时利用代理IP和多线程提升爬取效率。
this指针如何使C++成员指针可调用
|
11天前
|
存储 安全 编译器
【C++航海王:追寻罗杰的编程之路】引用、内联、auto关键字、基于范围的for、指针空值nullptr
【C++航海王:追寻罗杰的编程之路】引用、内联、auto关键字、基于范围的for、指针空值nullptr
36 5
|
10天前
|
C++ 容器
【编程技巧】 C++11智能指针
C++11引入了智能指针以自动管理内存,防止内存泄漏和悬挂指针: - `shared_ptr`:引用计数,多所有权,适用于多个对象共享资源。 - `unique_ptr`:独占所有权,更轻量级,适用于单一对象所有者。 - `weak_ptr`:弱引用,不增加引用计数,解决`shared_ptr`循环引用问题。 ## shared_ptr - 支持引用计数,所有者共同负责资源释放。 - 创建方式:空指针、new操作、拷贝构造/移动构造,以及自定义删除器。 - 提供`operator*`和`operator->`,以及`reset`、`swap`等方法。 ## unique_ptr
184 0
|
13天前
|
存储 Java C#
C++语言模板类对原生指针的封装与模拟
C++|智能指针的智能性和指针性:模板类对原生指针的封装与模拟
|
11天前
|
C++
C++职工管理系统(类继承、文件、指针操作、中文乱码解决)
C++职工管理系统(类继承、文件、指针操作、中文乱码解决)
13 0
C++职工管理系统(类继承、文件、指针操作、中文乱码解决)
|
13天前
|
设计模式 C++ 开发者
C++一分钟之-智能指针:unique_ptr与shared_ptr
【6月更文挑战第24天】C++智能指针`unique_ptr`和`shared_ptr`管理内存,防止泄漏。`unique_ptr`独占资源,离开作用域自动释放;`shared_ptr`通过引用计数共享所有权,最后一个副本销毁时释放资源。常见问题包括`unique_ptr`复制、`shared_ptr`循环引用和裸指针转换。避免这些问题需使用移动语义、`weak_ptr`和明智转换裸指针。示例展示了如何使用它们管理资源。正确使用能提升代码安全性和效率。
17 2
|
19天前
|
安全 数据库连接 C++
C++一分钟之-C++中的常量与宏定义
【6月更文挑战第18天】**C++编程中,常量(`const`和`constexpr`)确保值不变,用于配置、计算和逻辑控制。宏定义是预处理阶段的文本替换,适用于条件编译和简化代码。注意`const`的正确使用,避免过度的`const_cast`。宏可能导致命名冲突、副作用和类型安全问题,应谨慎使用。优选类型安全的常量定义,以提升代码质量和可维护性。**
28 5
|
18天前
|
存储 算法 安全
C++一分钟之-数组与指针基础
【6月更文挑战第19天】在C++中,数组和指针是核心概念,数组是连续内存存储相同类型的数据,而指针是存储内存地址的变量。数组名等同于指向其首元素的常量指针。常见问题包括数组越界、尝试改变固定大小数组、不正确的指针算术以及忘记释放动态内存。使用动态分配和智能指针可避免这些问题。示例代码展示了安全访问和管理内存的方法,强调了实践的重要性。
29 3
|
22天前
|
编译器 Linux C++
C++智能指针
**C++智能指针是RAII技术的体现,用于自动管理动态内存,防止内存泄漏。主要有三种类型:已废弃的std::auto_ptr、不可复制的std::unique_ptr和可共享的std::shared_ptr。std::unique_ptr通过禁止拷贝和赋值确保唯一所有权,而std::shared_ptr使用引用计数来协调多个指针对同一资源的共享。在C++17中,std::auto_ptr因设计缺陷被移除。**