【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




相关文章
|
3月前
|
C语言
【c语言】指针就该这么学(1)
本文详细介绍了C语言中的指针概念及其基本操作。首先通过生活中的例子解释了指针的概念,即内存地址。接着,文章逐步讲解了指针变量的定义、取地址操作符`&`、解引用操作符`*`、指针变量的大小以及不同类型的指针变量的意义。此外,还介绍了`const`修饰符在指针中的应用,指针的运算(包括指针加减整数、指针相减和指针的大小比较),以及野指针的概念和如何规避野指针。最后,通过具体的代码示例帮助读者更好地理解和掌握指针的使用方法。
68 0
|
1月前
|
存储 NoSQL 编译器
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
指针是一个变量,它存储另一个变量的内存地址。换句话说,指针“指向”存储在内存中的某个数据。
103 3
【C语言】指针的神秘探险:从入门到精通的奇幻之旅 !
|
1月前
|
存储 编译器 C语言
【C语言】指针大小知多少 ?一场探寻C语言深处的冒险 !
在C语言中,指针的大小(即指针变量占用的内存大小)是由计算机的体系结构(例如32位还是64位)和编译器决定的。
85 9
|
1月前
|
安全 程序员 C语言
【C语言】指针的爱恨纠葛:常量指针vs指向常量的指针
在C语言中,“常量指针”和“指向常量的指针”是两个重要的指针概念。它们在控制指针的行为和数据的可修改性方面发挥着关键作用。理解这两个概念有助于编写更安全、有效的代码。本文将深入探讨这两个概念,包括定义、语法、实际应用、复杂示例、最佳实践以及常见问题。
52 7
|
2月前
|
存储 C语言
C语言如何使用结构体和指针来操作动态分配的内存
在C语言中,通过定义结构体并使用指向该结构体的指针,可以对动态分配的内存进行操作。首先利用 `malloc` 或 `calloc` 分配内存,然后通过指针访问和修改结构体成员,最后用 `free` 释放内存,实现资源的有效管理。
215 13
|
2月前
|
存储 C语言 开发者
C 语言指针与内存管理
C语言中的指针与内存管理是编程的核心概念。指针用于存储变量的内存地址,实现数据的间接访问和操作;内存管理涉及动态分配(如malloc、free函数)和释放内存,确保程序高效运行并避免内存泄漏。掌握这两者对于编写高质量的C语言程序至关重要。
76 11
|
2月前
|
存储 程序员 编译器
C 语言数组与指针的深度剖析与应用
在C语言中,数组与指针是核心概念,二者既独立又紧密相连。数组是在连续内存中存储相同类型数据的结构,而指针则存储内存地址,二者结合可在数据处理、函数传参等方面发挥巨大作用。掌握它们的特性和关系,对于优化程序性能、灵活处理数据结构至关重要。
|
2月前
|
算法 C语言
C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项
本文深入讲解了C语言中的文件操作技巧,涵盖文件的打开与关闭、读取与写入、文件指针移动及注意事项,通过实例演示了文件操作的基本流程,帮助读者掌握这一重要技能,提升程序开发能力。
179 3
|
2月前
|
存储 算法 程序员
C 语言指针详解 —— 内存操控的魔法棒
《C 语言指针详解》深入浅出地讲解了指针的概念、使用方法及其在内存操作中的重要作用,被誉为程序员手中的“内存操控魔法棒”。本书适合C语言初学者及希望深化理解指针机制的开发者阅读。
|
2月前
|
程序员 C语言
C语言中的指针既强大又具挑战性,它像一把钥匙,开启程序世界的隐秘之门
C语言中的指针既强大又具挑战性,它像一把钥匙,开启程序世界的隐秘之门。本文深入探讨了指针的基本概念、声明方式、动态内存分配、函数参数传递、指针运算及与数组和函数的关系,强调了正确使用指针的重要性,并鼓励读者通过实践掌握这一关键技能。
57 1