【C语言】C语言之深入指针进阶(建议收藏以备不时之需)

简介: 【C语言】C语言之深入指针进阶(建议收藏以备不时之需)

image.png

【C语言】C语言——指针进阶

🗽初阶指针请点击此跳转

⭐1. 字符指针

⭐2.指针数组

⭐ 3. 数组指针

⭐4. 数组传参和指针传参

⭐ 5. 函数指针

⭐ 6. 函数指针数组

⭐7. 指向函数指针数组的指针

⭐8. 回调函数

🗽初阶指针请点击此跳转

https://blog.csdn.net/DerrickWestbrook/article/details/117787490.

指针的主题,在上述链接文章中已经解释过了,我们已经知道了指针的概念:


指针就是个变量,用来存放地址,地址唯一标识一块内存空间。

指针的大小是固定的4/8个字节(32位平台/64位平台)。

指针是有类型的,指针的类型决定了指针的±整数的步长,指针解引用操作的时候的权限。

在本篇文章将继续讲解质指针的高级主题


⭐1. 字符指针

指针里有一种指针类型为字符指针,字符指针存放一个字符的地址


image.png

除此之外还有一种使用方式,如:

image.png

值得注意的是

char* pstr = “Rip Kobe”;

这行代码的意思是把一个字符串“Rip Kobe”放到pstr指针变量里了吗?

其实不然,这行代码的本质是把字符串 Rip Kobe 的首字符也就是 R 的地址放到了pstr中,当需要用的时候,只需要找到这个字符串的首地址,就可以根据首地址找到整个字符串的地址

就像这行代码

printf("%s\n", pstr);

pstr中存的是字符串的首地址,printf函数就可以根据这个首地址把整个字符串打印出来


还有一个点需要注意

看下面这段代码

image.png

运行结果是

image.png

这里str3和str4指向的是一个同一个常量字符串。C/C++会把常量字符串存储到单独的一个内存区域当几个指针,指向同一个字符串的时候,他们实际会指向同一块内存。但是用相同的常量字符串去初始化不同的数组的时候就会开辟出不同的内存块。

所以str1和str2不同,str3和str4不同。


⭐2.指针数组

指针数组顾名思义就是一个存放指针的数组。

image.png



比如 int* arr1[10] 可以分开来看,

arr1[10]是一个有十个元素的数组,里面存放的元素类型为整型指针


⭐ 3. 数组指针

数组指针是指针?还是数组?

答案是:指针。

我们已经熟悉:

整形指针: int * pint; 能够指向整形数据的指针。

浮点型指针: float * pf; 能够指向浮点型数据的指针。

那数组指针应该是:能够指向数组的指针

image.png

这两行代码区别在于

第一行代码: * 先和int相结合,说明p1数组里存有十个整型指针,这是个指针数组

第二行代码: p先和 * 结合 说明p是一个指针变量,然后指针指向的是一个大小为10个整型的数组。

所以p是一个指针,指向一个数组,叫数组指针。

这里要注意:[ ]的优先级要高于 * 号的,所以必须加上()来保证p先和 * 结合。


&数组名VS数组名

对于下面的数组



image.png

arr 和 &arr 分别是什么呢?

我们知道arr是数组名,数组名表示数组首元素的地址。

&arr数组名到底是啥?

我们看一段代码:

image.png

运行结果如下:

image.png

可见数组名和&数组名打印的地址是一样的

为什么是一样的呢?

再来看如下代码

image.png

运行结果如下:

image.png

根据上面的代码我们发现,其实&arr和arr,虽然值是一样的,但是意义应该不一样的。

实际上: &arr 表示的是数组的地址,而不是数组首元素的地址,因为内存中访问数组,默认从首元素地址开始访问,所以表示整个数组的地址也是首元素的地址,但是意义是不同的。(细细体会一下)

数组的地址+1,跳过整个数组的大小,所以 &arr+1 相对于 &arr 的差值是40


接下来说说数组指针的使用

既然数组指针指向的是数组,那数组指针中存放的应该是数组的地址。

看代码:



image.png

上边这种写法一般很少用,数组指针多用于二维数组传参

请看如下代码:

image.png

运行结果如下:

image.png

这里需要注意几个点:

1.数组名arr,表示首元素的地址

2.但是二维数组的首元素是二维数组的第一行

3.所以这里传递的arr,其实相当于第一行的地址,是一个一维数组的地址

4.所以可以用数组指针来接收

但是接收的时候需要注意的是 int(*arr)[5] 和 int arr[3][5] 这两种写法都是可以的


指针数组和数组指针的总结:

int arr[5]; arr是一个整型数组,有5个int型元素

int *parr1[10]; parr1是一个数组,里面有10个元素,每个元素是一个指向int型的指针

int (*parr2)[10]; parr2是一个数组指针,该指针指向的数组有10个元素,每个元素是int型

int (*parr3[10])[5]; parr3是一个数组,里面有10个元素 ,每个元素是一个数组指针,该指针指向的数组有5个元素,每个元素的类型是int型


⭐4. 数组传参和指针传参

一维数组传参

可以通过数组,也可以通过指针来接收,几种方法方式如下:


image.png

①用数组来接收

image.png

②形参[]里的值可以省略,因为在这里没有意义因为传入的实际上是一个地址

image.png

③用整型指针接收一个整型数组的首元素地址

image.png

④用指针数组来接收指针,同理[ ]里的值可以省略,因为在这里没有意义

image.png

⑤这里的意思是arr2[20]里每个元素是int*

然后又把arr2的首元素地址传入了函数,所以要用 **arr 来接收;

image.png

二维数组传参

同理也可以通过数组或者指针来接收:

image.png

①用数组

image.png

②这种方式不可以,行可以省略但是列不可以省略,和初始化二维数组是一样的

image.png

③可以像这样省略行

image.png

总结:二维数组传参,函数形参的设计只能省略第一个[ ]的数字。
因为对一个二维数组,可以不知道有多少行,但是必须知道一行多少元素。
这样才方便运算。

④这种方式错误,二维数组的首元素是第一行,传入的其实是一个一维数组的地址,不能用一个整型指针来接收

image.png

⑤也不能这样,这是个整型指针数组

image.png

二维数组的首元素是第一行,传入的其实是一个一维数组的地址,一个一维数组的地址要用数组指针来接收,表示传入一个地址,这个地址指向一个有5个元素数组,每个元素是int型

image.png

⑦这种方式错误,不能在此处用二级指针

image.png

一级指针传参

image.png

思考:

当一个函数的参数部分为一级指针的时候,函数能接收什么参数?

比如:


image.png

答案如下:

image.png

二级指针传参

image.png

思考:

当函数的参数二级指针的时候,可以接收什么参数?

答案如下:

image.png

⭐ 5. 函数指针

函数指针变量——存放函数的地址


了解函数指针之前先来了解下函数的地址:

如图所示,打印函数 test 和 &test 得到的结果一模一样


这里需要注意一个问题:

函数名 和 &函数名 不同于 数组名 和 &数组名

数组名 和 &数组名,结果一样,但是意义不同

函数名 和 &函数名,结果和意义都一样,表示的都是函数的地址

image.png

函数名——函数的地址

&函数名——函数的地址

两种表示方式完全一样,都表示函数的地址

———————————————————————————————

函数指针的语法

image.png

这里的pf就是一个函数指针变量

这个函数指针类型int(*)(int,int)

表示指向参数是两个int,整个函数返回值类型为int的函数

函数指针的使用:


image.png

打印结果如下:

image.png

注意:

(*pf) (4,5) 也可以省略 * 号,写成 pf (4,5) 也是正确的,解引用符号只是为了方便理解指针,其实在此处解引用符号没有意义

接下来看一行代码加深理解:

image.png

这行代码要如何理解?

image.png


⭐ 6. 函数指针数组

函数指针数组——存放函数指针数组

语法如下:

image.png

函数指针数组用途:转移表

举例:计算器

image.png

⭐7. 指向函数指针数组的指针

指向函数指针数组的指针是一个指针 指针指向一个 数组 ,数组的元素都是函数指针

定义语法如下:

image.png

⭐8. 回调函数

回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。


举个栗子:

使用回调函数,模拟实现qsort(采用冒泡的方式)

注意:这里第一次使用 void * 的指针,讲解 void * 的作用

image.png




相关文章
|
1月前
|
存储 C语言
【C语言篇】深入理解指针3(附转移表源码)
【C语言篇】深入理解指针3(附转移表源码)
35 1
|
15天前
|
C语言
【c语言】指针就该这么学(1)
本文详细介绍了C语言中的指针概念及其基本操作。首先通过生活中的例子解释了指针的概念,即内存地址。接着,文章逐步讲解了指针变量的定义、取地址操作符`&`、解引用操作符`*`、指针变量的大小以及不同类型的指针变量的意义。此外,还介绍了`const`修饰符在指针中的应用,指针的运算(包括指针加减整数、指针相减和指针的大小比较),以及野指针的概念和如何规避野指针。最后,通过具体的代码示例帮助读者更好地理解和掌握指针的使用方法。
41 0
|
14天前
|
C语言
【c语言】指针就该这么学(3)
本文介绍了C语言中的函数指针、typedef关键字及函数指针数组的概念与应用。首先讲解了函数指针的创建与使用,接着通过typedef简化复杂类型定义,最后探讨了函数指针数组及其在转移表中的应用,通过实例展示了如何利用这些特性实现更简洁高效的代码。
11 2
|
15天前
|
C语言
如何避免 C 语言中的野指针问题?
在C语言中,野指针是指向未知内存地址的指针,可能引发程序崩溃或数据损坏。避免野指针的方法包括:初始化指针为NULL、使用完毕后将指针置为NULL、检查指针是否为空以及合理管理动态分配的内存。
|
15天前
|
C语言
C语言:哪些情况下会出现野指针
C语言中,野指针是指指向未知地址的指针,通常由以下情况产生:1) 指针被声明但未初始化;2) 指针指向的内存已被释放或重新分配;3) 指针指向局部变量,而该变量已超出作用域。使用野指针可能导致程序崩溃或不可预测的行为。
|
21天前
|
存储 C语言
C语言32位或64位平台下指针的大小
在32位平台上,C语言中指针的大小通常为4字节;而在64位平台上,指针的大小通常为8字节。这反映了不同平台对内存地址空间的不同处理方式。
|
21天前
|
存储 算法 C语言
C语言:什么是指针数组,它有什么用
指针数组是C语言中一种特殊的数据结构,每个元素都是一个指针。它用于存储多个内存地址,方便对多个变量或数组进行操作,常用于字符串处理、动态内存分配等场景。
|
21天前
|
存储 C语言
C语言指针与指针变量的区别指针
指针是C语言中的重要概念,用于存储内存地址。指针变量是一种特殊的变量,用于存放其他变量的内存地址,通过指针可以间接访问和修改该变量的值。指针与指针变量的主要区别在于:指针是一个泛指的概念,而指针变量是具体的实现形式。
|
22天前
|
C语言
C语言指针(3)
C语言指针(3)
11 1
|
22天前
|
C语言
C语言指针(2)
C语言指针(2)
12 1