前言

紧跟上一章,这一章的指针相对较难一些,不过经过本章的学习,相信大家能够有所得

一.二级指针

  • 上一章说的都是一级指针的应用,那么什么是二级指针呢?
  • 我们都知道,创建一个变量是需要在内存中开辟一个地址空间来存放的,那么指针(指针就是指针变量)也是如此,他也是一个变量,他也需要一个地址内存空间来存放,只不过这个空间放的是其他变量的地址。
  • 实际上,二级指针可以理解为就是指针的指针,也就是说,二级指针里放的是一级指针的地址
  • 我们创建一个整型变量给个初始值,再将这个变量的地址交给一个指针变量,然后将这个指针的地址再传给另一个指针,那么这个指针就叫做二级指针

我们看以下代码:

	int a = 10; // 创建一个变量a,他需要在内存中占有一段空间
	int* pa = &a; // pa为指针变量,他也需要一段空间来存放,但他里面放的是a的地址
	int** ppa = &pa; // 同理,它里面放的是pa这个指针的地址

在这里插入图片描述 在这里插入图片描述

》》 那么我们如何使用二级指针呢?

  • 由上述代码,我们都知道,对一级指针解引用操作可以找到a的值,并且能够修改a里面的值,同样的。对ppa解引用就能找到a的地址,因为指针变量pa存放的是a的地址,如下:
#include <stdio.h>

int main()
{
	int a = 10;
	int* pa = &a;
	int** ppa = &pa;

	printf("%p\n", &a);
	printf("%p\n", pa);
	printf("%d\n", *pa);

	printf("%p\n", &pa);
	printf("%p\n", ppa); // ppa就是pa的地址
	printf("%p\n", *ppa); // 对ppa解引用找到的是a的地址,因为pa里面存放的是a的地址

	return 0;
}

在这里插入图片描述

  • 也就是说通过对ppa两次的解引用,我们可以找到a的值:

在这里插入图片描述

  • ppa的解引用过程:一次解引用找到的是a的地址,因为ppa是指向int 类型的指针,他指向int型的指针变量pa,而pa里面存放的是a的地址。二次解引用便通过a的地址找到a的值。

二.指针数组

  • 指针数组他是数组还是指针呢???
  • 当然是数组。
  • 数组里的元素放的是指针,那这个数组就是指针数组。

例如:

int* arr[3] = {0};

arr是一个指针数组,他里面放了三个指针,每个指针是指向int类型的指针。

那么我们如何使用二维数组,下面展示一个比较能够体现其作用的代码:

#include <stdio.h>

int main()
{
	int a[4] = { 1,2,3,4 };
	int b[4] = { 5,6,7,8 };
	int c[4] = { 9,10,11,12 };
	int* arr[3] = { a,b,c };

	int i = 0;
	int j = 0;

	for (i = 0; i < 3; i++)
	{
		for (j = 0; j < 4; j++)
		{
			printf("%-2d ", *(arr[i] + j)); // arr[i][j] ==  *(arr[i] + j)
		}
		printf("\n");
	}

	return 0;
}
  • 可以看到,指针数组arr里面有三个指针,他们分别指向数组a的首元素地址,数组b的首元素地址,数组c的首元素地址,后面我们相继打印每个数组的内容,我们利用 *(arr[i] + j) 来寻找到每个数组对应下标的内容,arr[i]表示找到指针数组对应下标的指针指向的首元素地址,+j表示指针偏移找到该数组后面的元素的地址,对整体解引用则为该指针对应的数组的对应下标的值。

总结

指针初阶就到这里,想必大家对指针的认识已有不少。我们都知道,灵活的利用指针可以大大提高程序的效率,所以这里我们需要针对性的练习来提高我们对指针的敏感度。