深入解剖指针篇(3)

个人主页(找往期文章) :我要学编程(ಥ_ಥ)-CSDN博客

目录

二级指针

指针数组

指针数组模拟二维数组

字符指针变量 

数组指针

数组指针初始化 

二维数组传参的本质 

函数指针

函数指针的使用

typedef关键字

函数指针数组 


二级指针

指针变量也是变量,是变量就有地址,那指针变量的地址存放在哪里?答案是存放在二级指针里头。 指针变量的地址存放在哪里?这句话的主语是地址,问的是地址存放在哪里?我们从前面的学习知识可以知道地址是存放在指针里。只不过这里用了指针变量这个定语来修饰罢了。

深入解剖指针篇(3)_第1张图片

上面这个图就是二级指针创建的流程图。

1. *ppa 通过对ppa中的地址进行解引用,这样找到的是 pa , *ppa 其实访问的就是 pa。

2. **ppa 先通过 *ppa 找到 pa ,然后对 pa 进行解引用操作, *pa ,那找到的是 a 。

至于这个**ppa的两颗*的理解:

深入解剖指针篇(3)_第2张图片

这个是通过分解成一级指针变量来理解的。

至于ppa+-整数,能访问几个字节,是取决于 int* 的 。这个是指针是4/8个字节。因此ppa+-整数,能访问4/8个字节。

指针数组

指针数组是指针还是数组? 我们类比⼀下,整型数组,是存放整型的数组,字符数组是存放字符的数组。 那指针数组呢?是存放指针的数组。

指针数组的每个元素都是用来存放地址(指针)。如下图:

深入解剖指针篇(3)_第3张图片

指针数组的每个元素是地址,又可以指向一块区域。 

指针数组模拟二维数组

 当我们想要打印二维数组的所有元素时,我们是使用下标引用操作符([ ])来实现的。现在学习了指针,那么可以用指针来实现吗?答案是可以的。我们先用一个数组来存放另外几个数组的地址,再通过地址来找到对应的数组,最后再通过打印一维数组的方法来实现。

#include 
void Print(int** p, int sz, int sz1)//arr的类型是int*
{
	int i = 0;
	for (i = 0; i < sz; i++)
	{
		int j = 0;
		for (j = 0; j < sz1; j++)
		{
			//printf("%d ", p[i][j]);//下标引用的方法
			printf("%d ", *(*(p + i) + j));//指针引用的方法
		}
		printf("\n");
	}
}
int main()
{
	int arr1[] = { 1,2,3,4,5 };
	int arr2[] = { 2,3,4,5,6 };
	int arr3[] = { 3,4,5,6,7 };
	int sz1 = sizeof(arr1) / sizeof(arr1[0]);
	int* arr[] = { arr1,arr2,arr3 };
	int sz = sizeof(arr) / sizeof(arr[0]);
	Print(arr, sz, sz1);
	return 0;
}

上述的代码模拟出二维数组的效果,实际上并非完全是二维数组,因为每一行并非是连续的。

字符指针变量 

在指针的类型中我们知道有一种指针类型为字符指针 char*。

一般使用方式:深入解剖指针篇(3)_第4张图片

要注意这个ch的内容不可更改,因为这个ch中存放的是一个常量字符串。常量字符串的内容不可更改。但是存放在数组里的字符串可以更改。即数组内容可以更改。

还有一种使用方式如下:深入解剖指针篇(3)_第5张图片

这个 str 中存放的是hello world 的首字符的h,而不是存放hello world 。因为说到底这个str是一个指针变量,存放的是一个地址:这个字符串的地址,然而这个字符串的地址起始就是h的地址,因此就存放的是h的地址。至于在打印这个字符串时,为什么不用解引用?其实是因为str指向的内容就是这个字符串。如果我们还去解引用的话,就会把这个首字符给打印出来。

例如:

深入解剖指针篇(3)_第6张图片

《剑指offer》中收录了一道和字符串相关的笔试题,我们一起来观摩⼀下:

#include 
int main()
{
	char str1[] = "hello bit.";
	char str2[] = "hello bit.";
	const char* str3 = "hello bit.";
	const char* str4 = "hello bit.";
	if (str1 == str2)
		printf("str1 and str2 are same\n");
	else
		printf("str1 and str2 are not same\n");

	if (str3 == str4)
		printf("str3 and str4 are same\n");
	else
		printf("str3 and str4 are not same\n");

	return 0;
}

答案如下:深入解剖指针篇(3)_第7张图片

我们思考一下是为什么? 首先,str1与str2都是一个地址,在内存中,不可能有两个一模一样的地址。就比如:我们在生活中点外卖,如果在地图上有两个一模一样的地址,那么外卖小哥怎么会知道送去哪一个地方呢?这就产生了错误。因此就打印 str1 and str2 are not same 。接下来就看str3与str4,它们都是一个字符指针,指向的也都是同一个字符串。而我们刚刚知道了这个字符指针存放的是字符串首元素的地址。这个字符串是同一个,那么它们的首元素地址也是一样的。即str3与str4都是指向这个地址,所以str3等于str4,打印 str3 and str4 are same 。(C/C++会把常量字符串存储到单独的一个内存区域,因此str3与str4都是指向这个内存区域)。

数组指针

数组指针是指向数组的指针。我们已经知道了整形指针: int * p;,存放的是整形变量的地址,能够指向整形数据的指针。那数组指针变量应该是:存放的应该是数组的地址,能够指向数组的指针变量。

深入解剖指针篇(3)_第8张图片

想要知道答案,首先就得明白 * 和 [ ] 这两个操作符的优先级。

深入解剖指针篇(3)_第9张图片C 运算符优先级 - cppreference.com   这个是运算符优先级和结合性的网址。

由此可知:上面一个是数组,存放整形指针变量的,因此称为指针数组。下面一个是指针(()使p与*先结合——>指针),指向的是一个数组,因此称为数组指针。既然知道是指针了,那么怎么解读这个指针呢?如图所示:深入解剖指针篇(3)_第10张图片

数组指针初始化 

数组指针是指向数组的指针,存放的是数组的地址,那怎么获得数组的地址呢?就是我们之前学习的 &数组名 。这个就是得到的整个数组的地址。深入解剖指针篇(3)_第11张图片

这里可能会有小伙伴有疑惑:这个数组指针的元素个数能不能省略? 答案是不能。因为我们在数组里学过可以省略,但是这个不是数组,而是指针。举例:

深入解剖指针篇(3)_第12张图片

二维数组传参的本质 

我们还没有学习指针之前,二维数组传参是这样的:

#include 
void Print(int arr[3][5], int row, int col)//用数组传参的方式接收
{
	int i = 0;
	for (i = 0; i < row; i++)
	{
		int j = 0;
		for (j = 0; j < col; j++)
		{
			printf("%d ", arr[i][j]);
		}
		printf("\n");
	}
}
int main()
{
	int arr[3][5] = { {1,2,3,4,5},{2,3,4,5,6},{3,4,5,6,7} };
	//打印数组
	Print(arr, 3, 5);
	return 0;
}

深入解剖指针篇(3)_第13张图片

我们这个是用形参的方式来接收的。如果用指针呢?我们知道数组名是首元素的地址,在二位数组中,我们学过把二位数组看成几个一维数组组成。那么就可以在一维数组的层面把二维数组中的一维数组看成一个一个的元素。那么就可以推出来,在一维数组的层面,二维数组的数组名是首元素的地址,也就是二维数组中第一个一维数组的地址。深入解剖指针篇(3)_第14张图片

#include 
void Print(int(*p)[5], int row, int col)
{
	int i = 0;
	for (i = 0; i < row; i++)
	{
		int j = 0;
		for (j = 0; j < col; j++)
		{
			printf("%d ", *(*(p + i) + j));
			//(p+i)中的p是整个(第一个)一维数组的地址,+1,跳过的是整个一维数组。
			//因为int(*)[5]是数组指针的类型,这个是+1,就是跳过的。
			//*p访问的是第一个数组,而*(*p+j)访问的就是第一个数组里的元素。
		}
		printf("\n");
	}
}
int main()
{
	int arr[3][5] = { {1,2,3,4,5},{2,3,4,5,6},{3,4,5,6,7} };
	//打印数组
	Print(arr, 3, 5);
	//数组名是首元素的地址,也就是整个(第一个)一维数组的地址。
	//整个(第一个)一维数组的地址要用数组指针来接收。
	return 0;
}

深入解剖指针篇(3)_第15张图片

有的小伙伴可能会疑惑为什么 int(*)[5]是数组指针的类型?这个我在数组知识点这篇文章写过,大家可以去看看。意味着二维数组传参本质上也是传递了地址,传递的是第一行这个一维数组的地址

总结:二维数组传参,形参的部分可以写成数组,也可以写成指针形式。

函数指针

什么是函数指针呢? 根据前面学习整型指针,数组指针的时候,我们可以类比关系,我们不难得出结论: 函数指针应该是用来存放函数地址的,未来通过地址能够调用函数的。我们可以先看看函数的地址是啥样?是不是和数组是一样的?

void test()
{
	printf("hehe\n");
}
int main()
{
	printf("%p\n", &test);
	return 0;
}

深入解剖指针篇(3)_第16张图片

看来&函数名是能够把函数的地址地址取出来,那么就看看这个函数名是否能代表函数的地址?深入解剖指针篇(3)_第17张图片 

如此看来,这个函数名也是代表函数的地址。

总结:函数的地址用两种方法可以取出:1.  &函数名   2.   函数名

既然能把函数的地址取出,就肯定要放到函数指针里头。怎么存放呢?深入解剖指针篇(3)_第18张图片

还可以写成这样:

深入解剖指针篇(3)_第19张图片

函数指针的使用

通过函数指针调用指针指向的函数。

#include 
int Add(int x, int y)
{
	return x + y;
}
int main()
{
	int a = 0;
	int b = 0;
	scanf("%d%d", &a, &b);
	int (*p)(int, int) = Add;
	int ret = (*p)(3,4);//相当于Add(3, 4);
	printf("%d\n", ret);
}

深入解剖指针篇(3)_第20张图片

然而我们会发现这个代码也是可以的:深入解剖指针篇(3)_第21张图片

这也就意味着这个 * 其实有没有都无所谓。这个函数指针的变量名就相当于这个函数名。

typedef关键字

 typedef 是用来类型重命名的,可以将复杂的类型,简单化。

比如,你觉得 unsigned int 写起来不方便,如果能写成 uint 就方便多了,那么我们可以使用:

typedef unsigned int uint//将无符号整型重新命名为uint,这个就代表无符号整型

如果是指针类型,能否重命名呢?其实也是可以的,比如,将 int* 重命名为 ptr_t ,这样写: 

typedef int* ptr_t

但是对于数组指针和函数指针稍微有点区别。比如我们有数组指针类型 int(*)[5] ,需要重命名为 parr_t ,那可以这样写:

typedef int(*parr_t)[5]; //新的类型名必须在*的右边

函数指针类型的重命名也是一样的,比如,将 void(*)(int) 类型重命名为 pf_t ,就可以这样写: 

typedef void(*pf_t)(int)///新的类型名必须在*的右边

函数指针数组 

函数指针数组是一个数组,用来存放函数指针的。那么函数指针的数组如何定义呢?深入解剖指针篇(3)_第22张图片

p先和 [ ] 结合,说明 p 是数组,数组的内容是什么呢? 是 int (*)(int) 类型的函数指针。 

好啦,这就是C语言深入解剖指针第三篇的全部内容了!下期见!

你可能感兴趣的:(C语言,c语言)