C语言(指针)6

embedded/2024/10/9 15:17:20/

                    Hi~!这里是奋斗的小羊,很荣幸各位能阅读我的文章,诚请评论指点,关注+收藏,欢迎欢迎~~     

                                💥个人主页:小羊在奋斗

                                💥所属专栏:C语言   

        本系列文章为个人学习笔记,在这里撰写成文一为巩固知识,二为同样是初学者的学友展示一些我的学习过程及心得。文笔、排版拙劣,望见谅。

                                一、函数与指针

                                                1.函数指针

                                                2.函数指针

                                                3.typedef 关键字

一、函数与指针

1.函数的地址

        我们知道,变量有地址,数组有地址,指针也有地址,那函数肯定也有地址。存放变量地址的指针叫一级指针,存放数组地址的指针叫数组指针,存放指针地址的指针叫二级指针,那存放函数的指针就叫函数指针。在学习函数指针之前,我们先来探讨一下函数的地址。

        可以看到,我们用 “&函数名” 的方式就能拿到函数的地址。我们学习了数组后知道,数组名表示的是数组首元素的地址,那函数名是不是也表示什么的地址呢?我们不用 “&” 操作符直接打印函数名的地址来看一下:

        居然也打印出了函数的地址。那跟数组相比,“&数组名” 是数组的地址,“数组名” 数数组首元素的地址,“&函数名” 和 “函数名” 是不是也类似呢?其实不是的,“&函数名” 和 “函数名” 表示的都是函数的地址,它们的效果是一样的,没有区别。 

2.函数指针

        在了解了函数的地址后,为了存放函数的地址,我们就来探究函数指针。跟其他指针一样,第一步肯定是要确定指针的类型。函数指针的类型跟数组指针的类型在形式上是非常相似的,这里就不卖关子直接给出来了, 然后我们再详细分解其中各部分的含义。

        去掉上面的函数指针变量名剩下的就是函数指针的类型,可以看到函数指针类型和数组指针类型很像,同样的,表示指针变量的 “ * ” 和指针变量名是结合在一起的,要用圆括号括起来。

        在监视窗口不仅能看到变量的值,也能看到变量的类型:

        我们将函数的地址存到函数指针变量中后,使用的方法和其他指针一样吗?是的,同样是用解引用操作符 “ * ” 解引用函数指针变量:

         同样的,解引用操作时 “ * ” 和指针变量名也要用圆括号括起来,不然指针变量名就会与后面的括号结合,使得指针变量名变成一个函数名。

        不用函数指针的时候我们函数调用是:函数名(x, y)。“&函数名” 和 “函数名” 都表示函数的地址,所以说 “指针变量名” 和 “函数名” 其实是等价的,那么,函数调用的写法:函数名(x,y

),不就等价于:指针变量名(x, y)嘛,所以可以得出的是用函数指针进行函数调用的时候可以省略解引用操作符 “ * ”:

        事实证明确实是可以的。

        在了解完上面的内容后,我们来看两个很有意思的代码: 

      示例(1):

        该如何理解上面的代码呢? 在看了上面函数指针的内容后,相信我们很容易就能看到 “ void (*)()” 这个函数指针类型,这个函数指针类型对应的函数的返回值是void,且没有参数。那既然是类型,那 “(类型)0 ” 表示的不就是强制类型转换嘛,所以 “ void (*)()0 ” 的意思就是将0强制类型转换为一个函数地址,那剩下的 “(*函数地址)()” 就是一个函数调用,没有参数。

        总结:这是一次函数调用,是将数字 “ 0 ” 用函数指针类型强制类型转换为一个函数地址,然后对 “ 0 ” 这个函数地址解引用,调用的是 “ 0 ” 地址处的那个函数。这个函数没有参数,没有返回值。另外,其中左边的第一个 “ * ” 可以省略,因为可以直接用 “ 地址()” 的形式进行函数调用。函数名和函数的地址都可以进行函数调用。

      示例(2):

        类比(1),我觉得我们需要找到一个突破口, 那你觉得突破口是什么呢?

        我们看到 “ signal ” 和 “ * ” 没有用圆括号括起来,那就说明 “ signal ” 和后面的圆括号结合了,那 “ signal ” 应该是一个函数名,而函数名后面圆括号内应该就是函数参数,参数有两个,第一个参数的类型是 int 型,第二个参数的类型是一种函数指针类型,该函数指针指向的函数的参数类型是int型,返回值是void型。

        对一个函数来说,我们讨论了它的函数名(上面是signal),讨论了它的参数,那还剩下它的返回值没有讨论,那么,对于上面的代码,除过 “ signal(int, void(*)(int))” 剩下的就是函数的返回值了,返回值类型还是一个函数指针类型,该函数的参数类型也是int型,返回值也是void型。

        我们知道对于一个函数而言无非就是函数定义、函数调用和函数声明,那上面的代码到底是哪种呢?函数定义的话必须要有函数体,很明显不是函数定义;那函数调用的话也没有参数(我们通过上面的代码已经知道signal函数是有参数的,但是只有参数类型没有参数值),所以也不是函数调用;那就只剩下函数声明了,所以上面的代码是一个函数声明

        但是我们之前见过的函数声明的格式是:函数的返回值类型 函数名(函数参数)。很明显上面的代码不符合我们现在的认知,因为它把 “函数名(函数参数)” 放到了函数返回值类型里面。虽然这让我们看上去很别扭,但语法规定就是这么写的。

3.typedef关键字

        在细细了解了上面这两个有意思的代码后,我们会觉得很复杂,原因是类型的形式很复杂,那我们有没有什么办法能简化一下这种复杂的类型呢?接下来我们介绍一个关键字:typedef。

        typedef 类型名 重定义的类型名; (注意后面有一个分号) 

        typedef 是用来类型重命名的,可以将复杂的类型简单化。比如我们可以将 “ unsigned int ” 简化为 “ uint ”,以后就可以用 “ uint ” 来代替 “ unsigned int ” 了。同样的,指针类型也是可以重命名的,比如我们将 “ int * ” 重命名为 “ p_i ”,将 “ char * ” 重命名为 “ p_c ” 等。

        同样的,数组指针类型和函数指针类型也都是可以用 typedef 关键字重命名的,但是对于数组指针和函数指针来说稍微有点区别。如果按照上面的写法,对数组指针类型和函数指针类型的重定义应该是下面这样:

       但事实是编译器报错。这是为什么呢?事实上应该写成下面这样:

        也就是说我们要把重定义的类型名写到原来数组指针变量名 / 函数指针变量名的位置 

        那既然有了 typedef 关键字,我们就可以简化一下上面示例(2)中的代码了。原代码是: void( *signal(int, void(*)(int)))(int)。我们知道 “ void (*)(int)” 是一个函数指针类型, “ signal ” 函数的返回值也是一个同样的函数指针类型,那我们就可以用 typedef 关键字来简化一下这个类型:

        当我们这样写后,这条代码表达的意思我们就能很直观地明白了。 还是之前那句话,随着学习的不断深入,我们写出的代码质量会越来越高。

           如果觉得我的文章还不错,请点赞、收藏 + 关注支持一下,我会持续更新更好的文章。


http://www.ppmy.cn/embedded/40907.html

相关文章

【GoLang基础】垃圾回收是如何工作的?

问题引出: Go语言中的垃圾回收是如何工作的? 解答: 在 Go 语言中,垃圾回收(Garbage Collection,GC)是自动管理内存的机制,用于在运行时识别和释放不再使用的内存对象,以…

理解安卓系统的三个时间

安卓设备有三种不同的可用时钟: System.currentTimeMillis()SystemClock.uptimeMillis()SystemClock.elapsedRealtime() 一、System.currentTimeMillis() System.currentTimeMillis()是一个标准的“墙”时钟(时间和日期),表示从纪元到现在的毫秒数。该…

Spring学习笔记

目录 1. Spring有什么优势 1.1 模块化 1.2 轻量级 1.3 方便集成各种优秀框架 1.4 提供了分层开发下的完整技术解决方案 1.5 Java语言编写的开源框架,使用了多种设计模式 2. Spring的第一个程序 2.1 开发环境 2.2 环境搭建 2.3 编码测试 2.4 BeanFactory的UML类图…

什么是HTTP/2?

HTTP/2(原名HTTP 2.0)即超文本传输协议第二版,使用于万维网。HTTP/2主要基于SPDY协议,通过对HTTP头字段进行数据压缩、对数据传输采用多路复用和增加服务端推送等举措,来减少网络延迟,提高客户端的页面加载…

springboot 注解(持续更新中)

RequestBody RequestBody将json格式的数据转为java对象(字段名称要一致) RequestBody主要用来接收前端传递给后端的json字符串中的数据的(请求体中的数据的);GET方式无请求体,所以使用RequestBody接收数据时,前端不能…

TypeScript学习笔记:入门指南

介绍 TypeScript 是一个由微软开发的开源编程语言,它是 JavaScript 的超集,添加了静态类型和面向对象的特性,使得 JavaScript 更加适合大型项目的开发。本文将介绍 TypeScript 的基本概念、特点以及其在实际项目中的作用。 特点 静态类型系…

Mysql 隔离级别

MySQL的事务隔离级别是指在处理并发事务时,为保证数据的一致性和事务的独立性,数据库系统提供的不同级别控制策略。根据ACID特性中的隔离性(Isolation),MySQL支持四种标准的事务隔离级别,每种级别有不同的并…

LeetCode 125题:验证回文串

❤️❤️❤️ 欢迎来到我的博客。希望您能在这里找到既有价值又有趣的内容,和我一起探索、学习和成长。欢迎评论区畅所欲言、享受知识的乐趣! 推荐:数据分析螺丝钉的首页 格物致知 终身学习 期待您的关注 导航: LeetCode解锁100…