(一)指针的定义

  • 一级指针:是一个指针变量,指向一个普通变量,并保存该普通变量的地址;
  • 二级指针:是一个指针变量,指向一个一级指针,并保存该一级指针的地址;

(二)引入二级指针

由于一级指针已经很熟悉,这里不再赘述,这里我们只谈谈二级指针
下面先简单使用一个二级指针看看

#include <stdio.h>

int main()
{
	int a = 10;
	int b = 20;
	int *p = &a;
	int** s = &p;

	//一次解引用*s 此时类型int*
	*s = &b;
	//二次解引用**s 此时类型int
	**s = 200;
	return 0;
}

逻辑关系如下:
a是一个int类型的变量,一级指针p指向a,并保存a的地址;
二级指针变量s指向一级指针变量p,并保存p的地址

二级指针s解引用操作:

  • 一次解引用
    *s的类型变成了(int*)(代表着一级指针p)间接改变了p的指向,从a的地址变成了b的地址;
  • 二次解引用
    s的类型变成了int (代表着变量b),此时s = 200;(等价于b = 200;)

(三)深入理解二级指针

(1)下面举个例子:
#include <stdio.h>

int main()
{
	//普通变量
	int a1 = 1;
	int a2 = 1;
	int a3 = 1;
	
	//一级指针
	int* p1 = &a1;
	int* p2 = &a2;
	int* p3 = &a3;
	
	//二级指针
	int **s = &p1; 
	return 0;
}

(假设a1,a2,a3空间连续,p1,p2,p3空间连续)逻辑图如下:
在这里插入图片描述
接下来我们结合上面的逻辑图看看下面这张表
在这里插入图片描述

分析:

  • s+1 表示二级指针s指向了p2,,移动的字节数需要根据指向的数据的空间大小进行计算sizeof(int*) * 1,所以移动4字节,此时s+1还是二级指针,所以类型int**
  • *s+1 先对s进行一次解引用为*s,相当于操控一级指针p1,然后*s + 1 ,相当于p1指向了a2的地址,所以移动了sizeof(int) * 1 = 4字节 ,此时的类型为 int*
  • **s + 1表示二次解引用,相当于a1的值+1,所以a1 = 2; a1的类型就是int
(2)在(1)的前提下,那要是把所有的类型换成char、short、double类型呢
  • char类型(1字节)

在这里插入图片描述

  • short类型(2字节)
    在这里插入图片描述

  • double类型(8字节)
    在这里插入图片描述

(四)总结

  • 在深入理解了二级指针的逻辑处理过程之后,我们不难发现:
  • 对二级指针变量s指针的移动操作时,s都会将以保存的一级指针的类型进行解析步长(s + sizeof( p) * n1),而一级指针*s(相当于p一级指针变量)会以保存的变量的类型进行解析步长(*s + sizeof(a) *n1

  1. n指的是移动的次数,本例题是n = 1 ↩︎ ↩︎

Logo

旨在为数千万中国开发者提供一个无缝且高效的云端环境,以支持学习、使用和贡献开源项目。

更多推荐