C语言易错知识点:二级指针、数组指针、函数指针

指针在C语言中非常关键,除开一些常见的指针用法,还有一些可能会比较生疏,但有时却也必不可少,本文章整理了一些易错知识点,希望能有所帮助!

1.二级指针:

2ab567f99dc74a5fb7c598652e9fe23a.png

parr是一个指针数组,其中每个元素存储字符串首字符的地址

d120d1e5c04f4abb89f9e27ac31d35c2.png

parr在这里表示首元素的地址,首元素是"Hello!"的'H'的地址,所以parr表示的是'H'的地址的地址,用char**来存储,如果解引用p就得到'H'的地址,相当于parr[0]。

如果使用p + 1,就会跳过一个char*的大小,在这里是8个字节(64位平台)。要理解p的加法含义,我们需要知道内存中的每一个字节都有一个地址,这个地址相当于内存的标签,它指向我们存储的内存空间,存储该数据需要k个字节,那么这些地址的数值差也是k。

83d0070198314e248fdfc435a742380e.png

以上面的例子讲解:p对应的是首元素地址,相当于一块内存空间的标签,在p这个地址下存储char*的具体值,占了8个字节。p + 1就会找到紧接着第一个char*的下一个char*存储的内存空间的地址。p + 2同理。

0ea1c3841d1048c09ac906b4f6733734.png

139c85e5aa354b77b3e3a2e33c152b3f.png

我们可以通过这两张图看到,地址是内存空间的标签,在内存里存储数据,地址则是负责找到这块空间。理解了这里,那么指针的大部分问题都会解决。

2.一维数组指针:

注意数组指针的本质是指针,它存储的值是数组的首元素地址,象征着它指向的对象是一个数组。但是,很多人会注意到,既然存储的值和单独取首元素地址的指针存储的值相同,那么怎么区分它们?

区分它们的主要方式有以下几点:首先,指针+1跳过的大小不同,数组指针+1会跳过数组的大小,而单独取首元素地址对应的指针+1只会跳过该元素的大小。

这里可以轻松算出两者之间相差32,即一个数组的大小。

注意:arr单独表示整个数组,而arr + 0 表示第一个元素地址,两者有本质区别

那么,既然两者存储的值相同,那么它们之间在除强制转换这种方法以外是否有其它转换方法?

我们知道,数组指针是对整个数组取地址,即&arr,假设这个指针表示为int (*parr) [10] = &arr,那么*parr得到的就是*(&arr)即arr,相当于首元素地址,也就是int*,所以首元素的地址也可表示为int* p = *parr。

我们要尝试理解它的本质,不要死记硬背,学会分析会帮助我们更好的运用指针。

下面四种写法均等效:

3.二维数组指针:

二维数组指针是一维数组指针的变形,理解了一维数组的指针,二维数组指针就可以轻松应对了。

首先熟悉一下二维数组的初始化过程:

其中arr[2][2]中第一个2代表2行,第二个2代表2列。

下面是有关二维数组指针易混的代码,可以尝试解释:

对数组的解释具体如下图:

理解了这里,那么二维数组就基本掌握了,下面看看整型数组指针和整型指针的转换,加深理解:

说到底,二维数组就是将一维数组作为元素存储在数组中,刚开始会觉得有点绕,不过多看几遍就能很好理解运用了。

注意二维数组传参:

这里arr传的是首元素(数组)地址,要用数组指针来接收,数组指针传参不能省略元素个数

二维数组在函数内取值要注意层级:

&arr是对整个arr取地址,是二维数组的指针,第一次解引用得到arr,首元素(数组)的地址,即一维数组指针,第二次解引用是对元素数组解引用,得到首元素(整型)的地址(注意理解)即int*,第三次解引用就能得到值。

4.函数指针:

函数指针一般用的不多,但是我们要清楚它的用法。对于int Fun (int a, int b),其对应的函数指针应表示为int (*pFun) (int a, int b) = Fun,当然也可写作&Fun。使用过程中,可以直接将pFun当作函数名一样使用,如pFun (1, 2)。多个函数指针也可以组成一个数组,成为函数指针数组,如int (*pFunarr[2]) (int a, int b) = { pFun1, pFun2 };它可以用于传递参数相同,且实现的功能有相似度的几种函数的集合,如加减乘除的函数的集合。


#include <stdio.h>

int add(int x, int y)
{
	return x + y;
}

int sub(int x, int y)
{
	return x - y;
}

int mul(int x, int y)
{
	return x * y;
}

int div(int x, int y)
{
	return x / y;
}

int main()
{
	int (*pFun[4]) (int x, int y) = { add, sub, mul, div };

	printf("%d\n%d\n%d\n%d", pFun[0](1, 1), pFun[1](1, 1), pFun[2](2, 3), pFun[3](8, 2));

	return 0;
}

至于这种数组的大小,就看函数参数的大小和数组的元素个数就能算了:

注意函数指针和函数指针数组传参:

 注意因为函数指针创建可以&fun也可直接fun,在解引用时也可选择加*或不加*

但是要注意层级不要弄错:

这里是函数指针的指针,pFun + 1相当于找到了存储减法函数地址的地址,但是只有在减法函数地址这个层级才能使用。

最近更新

  1. docker php8.1+nginx base 镜像 dockerfile 配置

    2024-03-22 04:58:01       94 阅读
  2. Could not load dynamic library ‘cudart64_100.dll‘

    2024-03-22 04:58:01       101 阅读
  3. 在Django里面运行非项目文件

    2024-03-22 04:58:01       82 阅读
  4. Python语言-面向对象

    2024-03-22 04:58:01       91 阅读

热门阅读

  1. 相向双指针

    2024-03-22 04:58:01       32 阅读
  2. RabbitMQ docker 单机部署

    2024-03-22 04:58:01       44 阅读
  3. vue绑定key

    2024-03-22 04:58:01       39 阅读
  4. AGI的数据驱动:挖掘海量信息的价值与智慧

    2024-03-22 04:58:01       45 阅读
  5. Mysql批量更新: on duplicate key update

    2024-03-22 04:58:01       45 阅读
  6. [蓝桥杯2012] 罗马数字

    2024-03-22 04:58:01       41 阅读
  7. sqllab第29-33通关笔记

    2024-03-22 04:58:01       44 阅读
  8. [AIGC] Apache HTTP服务器:历史与使用

    2024-03-22 04:58:01       37 阅读