C语言(指针)6

                    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.hqwc.cn/news/697453.html

如若内容造成侵权/违法违规/事实不符,请联系编程知识网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

Android 几种系统升级方式详解

目录 ◆ 概述 ● 几种启动模式 ● MISC分区 ● CACHE分区 ● 几种系统升级方式 ◆ Recovery升级 ● 升级包构成,签名,制作 ● 升级脚本 ● 升级过程 ◆ OTA升级 ● 升级包构成,制作 ● 升级脚本 ● 升级过程 ◆ fastboot升级 ◆ ADB升级 几…

【数据结构陈越版笔记】第1章 概述【习题】

1. 碎碎念 我这答案做的可能不对,如果不对,欢迎大家指出错误 2. 答案 1.1 判断正误 (1) N ( log N ) 2 N(\text{log}N)^{2} N(logN)2是 O ( N 2 ) O(N^{2}) O(N2)的。 (2) N 2 ( log N ) 2 N^{2}(\text…

Weblogic < 10.3.6 ‘wls-wsat‘ XMLDecoder 反序列化漏洞(CVE-2017-10271)

1 漏洞概述 CVE-2017-10271 是一个存在于 Oracle WebLogic Server 10.3.6 以下版本中的 XMLDecoder 反序列化漏洞。此漏洞源于 WebLogic 的 WLS-WebServices 核心组件,该组件使用 XMLDecoder 来解析用户传入的 XML 数据。由于 XMLDecoder 在处理某些特定格式的 XML…

Recommender ~ Collaborative filtering

Using per-item features User j 预测 movie i: Cost Function: 仅求和用户投票过的电影。 常规规范化(usual normalization):1/2m 正则化项:阻止过拟合 在知晓X的前提下,如何学习w,b参数…

Redis实战笔记

黑马点评项目笔记 一:数据交互: 1.把String解析成Java对象集合并且存入Redis及Java对象集合转换成JSON。 Overridepublic Result queryTypeList() {String s stringRedisTemplate.opsForValue().get("cache:list:");System.out.println(&qu…

第187题| 快速学会“阿贝尔定理”| 无穷级数(十五)|武忠祥老师每日一题

解题思路:这道题没有告诉我们是多少,没办法求出收敛半径,所以我们只能根据题目给的两个条件来解题(选项代入法)。 1.x-1,说明收敛的中心点是1,观察下列选项,显然答案在C和D之中。 …

腐烂的橘子 - (LeetCode)

一、概述 994. 腐烂的橘子 - 力扣(LeetCode),今天刷到这道题,开始按照自己实现的思路写了一次,通过了调试,但是提交的时候,来了一个大的数据,就没有通过测试,百思不得其…

HIVE解决连续登录问题

HIVE解决连续登录问题 目录 HIVE解决连续登录问题 1.解决连续登录问题 如何去分析数据: 2.需求: 3.-- 间隔天数 1.解决连续登录问题 如何去分析数据: 1)查看数据的字段信息 …

【简单介绍下Milvus】

🌈个人主页: 程序员不想敲代码啊 🏆CSDN优质创作者,CSDN实力新星,CSDN博客专家 👍点赞⭐评论⭐收藏 🤝希望本文对您有所裨益,如有不足之处,欢迎在评论区提出指正,让我们共…

01基础篇

1、初识 JVM 1.1 什么是 JVM JVM 全称是 Java Virtual Machine,中文译名 Java虚拟机。JVM 本质上是一个运行在计算机上的程序,他的职责是运行Java字节码文件。 Java源代码执行流程如下: 分为三个步骤: 编写Java源代码文件。使用…

数字化信息协同助力智能巡查,基于YOLOv5全系列【n/s/m/l/x】参数模型开发构建无人机数字侦查场景下智能靶标检测识别系统

无人机的快速发展与普及,使得其进入千家万户各行各业,发挥着越来越重要的作用。随着科技的飞速发展,未来的数字信息化战场正逐渐展现出其独特的作战形态。在这个以数据和信息为主导的新战场上,无人机侦查手段与人工智能目标智能检…

template——模板进阶(C++)

在之前的文章中,介绍了模板初阶:Cpp_桀桀桀桀桀桀的博客-CSDN博客 在本篇中将会对模板进一步的讲解。本篇中的主要内容为:非类型模板参数、函数模板的特化、类模板的特化(其中包含全特化和偏特化),最后讲解…