C++中 二重指针详解

2024-06-11 20:18
文章标签 c++ 指针 详解 二重

本文主要是介绍C++中 二重指针详解,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

前言
上篇博文讲到数组和指针的一些关系, 本篇博文会在这基础之上讲解一些更为复杂的指针, 这些指针在笔试题中经常出现, 对刚毕业找工作的大学生有很大的作用。

指向指针的指针
在指针篇(1)中, 我们所定义的指针指向一个一般的 int 类型变量, 但实际上指针完全可以指向另外一个指针, 而这个被指向的指针可以指向一个一般变量, 是不是很拗口? 在 C 语言当中, 这个指针被称为指针的指针, 也叫作二重指针。 在面试题目中就会叫你定义一个「指向指针的的指针,该指针指向的指针指向一个整型数(A pointer to a pointer to an integer)」。

假设该指针变量名为「 p2 」, 那么该二重指针就可以定义为「 int **p2」。 下面我们用一个具体的代码来说明怎么定义使用一个二重指针:

/* 
    GCC 5.4 环境编译 
     
*/  
  
#include <stdio.h>  
  
int main(void)  
{  
    int count = 5;            //局部变量的声明和初始化,在使用该局部变量前必须赋初值  
    int *p = &count;         //指针的声明和初始化,建议指针在声明时就给一个明确的指向,或者等于NULL                   
    int **p2 = &p;             //二重指针的声明和初始化
    
     
//  printf("count = %d\n",*count);   // 编译时会出现error: invalid type argument of `unary *'   
    printf("count = %d\n",count);    //一般变量的直接访问   
    printf("&count = %p\n",&count);  //一般变量的取地址   
      
      
    printf("&p = %p\n",&p);         //指针变量的取地址   
    printf("p = %p\n",p);           //指针变量的直接访问   
    printf("*p = %d\n",*p);         //指针变量的间接访问   
    
    
    printf("&p2 = %p\n",&p2);       //二重指针变量的取地址   
    printf("p2 = %p\n",p2);         //二重指针变量的直接访问   
    printf("*p2 = %p\n",*p2);       //二重指针变量的一次解引用间接访问   
    printf("**p2 = %d\n",**p2);     //二重指针变量的二次解引用间接访问 
    
    return 0;  
}
输出结果如下所示:

count = 5
&count = 0x7ffc7dfe5194
&p = 0x7ffc7dfe5198
p = 0x7ffc7dfe5194
*p = 5
&p2 = 0x7ffc7dfe51a0
p2 = 0x7ffc7dfe5198
*p2 = 0x7ffc7dfe5194
**p2 = 5

我们在结果中可以发现几个相同量:

count  = *p = **p2

&count = p = *p2

&p = p2

这几个量为什么相同? 如果你能自己分析清楚里面的关系, 那么你对 C 语言的了解就不再停留在初学阶段了。 为了更容易解释, 我们来看下面这张图:

 

图中的小矩阵内分别为整形变量 count、 一重指针 p、 二次指针 p2 的内容(值); 椭圆内分别为对应变量的地址(使用「&」取址)。 箭头上的「*」是对该变量进行解引用运算, 箭头指向变量解引用后的结果。 读懂该图后图中的所有变量关系就一目了然了:

指针 p2 中存放的是指针 p 的地址, 指针 p 中存放的是 count 的地址。
指针 p2 一次解引用后得到指针 p 的值, 指针 p 解引用后得到 count 的值。
指针 p2 二次解引用后得到 count 的值。 
值得一提的是二重指针虽然可以直接二次解引用得到 count 的值, 但是无法直接指向 count,也就是说「 int **p2 = &count 」这种写法是错误的, 这个从类型不匹配的角度也可以理解。

在《C 语言返璞归真: 指针篇(2)》中说到了指针和数组的关系, 知道一重指针可以指向一维数组。 扩展一下思维, 既然指针有二重, 相应的数组也会有二维。那么二重指针是否可以指向二维数组呢? 答案是不可以, 这是一个比较容易犯的错误。 二维数组名在赋值运算符右侧时会隐式转换为「int (*)[]」类型。 这种类型的指针被称为数组指针。
 

这篇关于C++中 二重指针详解的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1052187

相关文章

Debezium 与 Apache Kafka 的集成方式步骤详解

《Debezium与ApacheKafka的集成方式步骤详解》本文详细介绍了如何将Debezium与ApacheKafka集成,包括集成概述、步骤、注意事项等,通过KafkaConnect,D... 目录一、集成概述二、集成步骤1. 准备 Kafka 环境2. 配置 Kafka Connect3. 安装 D

Java中ArrayList和LinkedList有什么区别举例详解

《Java中ArrayList和LinkedList有什么区别举例详解》:本文主要介绍Java中ArrayList和LinkedList区别的相关资料,包括数据结构特性、核心操作性能、内存与GC影... 目录一、底层数据结构二、核心操作性能对比三、内存与 GC 影响四、扩容机制五、线程安全与并发方案六、工程

C++初始化数组的几种常见方法(简单易懂)

《C++初始化数组的几种常见方法(简单易懂)》本文介绍了C++中数组的初始化方法,包括一维数组和二维数组的初始化,以及用new动态初始化数组,在C++11及以上版本中,还提供了使用std::array... 目录1、初始化一维数组1.1、使用列表初始化(推荐方式)1.2、初始化部分列表1.3、使用std::

C++ Primer 多维数组的使用

《C++Primer多维数组的使用》本文主要介绍了多维数组在C++语言中的定义、初始化、下标引用以及使用范围for语句处理多维数组的方法,具有一定的参考价值,感兴趣的可以了解一下... 目录多维数组多维数组的初始化多维数组的下标引用使用范围for语句处理多维数组指针和多维数组多维数组严格来说,C++语言没

Spring Cloud LoadBalancer 负载均衡详解

《SpringCloudLoadBalancer负载均衡详解》本文介绍了如何在SpringCloud中使用SpringCloudLoadBalancer实现客户端负载均衡,并详细讲解了轮询策略和... 目录1. 在 idea 上运行多个服务2. 问题引入3. 负载均衡4. Spring Cloud Load

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

在 Spring Boot 中使用 @Autowired和 @Bean注解的示例详解

《在SpringBoot中使用@Autowired和@Bean注解的示例详解》本文通过一个示例演示了如何在SpringBoot中使用@Autowired和@Bean注解进行依赖注入和Bean... 目录在 Spring Boot 中使用 @Autowired 和 @Bean 注解示例背景1. 定义 Stud

如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解

《如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别详解》:本文主要介绍如何通过海康威视设备网络SDK进行Java二次开发摄像头车牌识别的相关资料,描述了如何使用海康威视设备网络SD... 目录前言开发流程问题和解决方案dll库加载不到的问题老旧版本sdk不兼容的问题关键实现流程总结前言作为

SQL 中多表查询的常见连接方式详解

《SQL中多表查询的常见连接方式详解》本文介绍SQL中多表查询的常见连接方式,包括内连接(INNERJOIN)、左连接(LEFTJOIN)、右连接(RIGHTJOIN)、全外连接(FULLOUTER... 目录一、连接类型图表(ASCII 形式)二、前置代码(创建示例表)三、连接方式代码示例1. 内连接(I

Go路由注册方法详解

《Go路由注册方法详解》Go语言中,http.NewServeMux()和http.HandleFunc()是两种不同的路由注册方式,前者创建独立的ServeMux实例,适合模块化和分层路由,灵活性高... 目录Go路由注册方法1. 路由注册的方式2. 路由器的独立性3. 灵活性4. 启动服务器的方式5.