前言
提示:相信大家在C语言的学习中一定会接触指针这样一个东西,而指针也是新手路上一定要消灭的boss,如果以后还要学习Java的同学更是要注重指针的学习。本文着重介绍函数指针,希望本文对屏幕前的你有所帮助!
提示:以下是本篇文章正文内容,下面案例可供参考
一、指针引子
示例:我们常常接触的指针大多有如下几类:
整形指针-存放整形地址,指向整形
字符指针-存放字符地址,指向字符
数组指针-存放数组地址(注意不是数组首元素地址),指向数组
由以上三个例子,我们能总结指针的共同点:存放某个类型变量的地址,指向那个类型的变量,但是在讲函数指针首先有一个问题:函数也有地址吗?我们用一段简单的代码来验证一下即可。
#include<stdio.h>
int Add(int x,int y)
{
return x+y;
}
int main()
{
printf("%p\n",&Add);
return 0;
}
屏幕上打印出地址:
所以答案是有的,函数也存在地址,那么也就衍生出了今天的知识点-函数指针。
二、使用步骤
1.取函数地址
我们知道&数组名,取出的是数组的地址。单独一个数组名,取出的是数组首元素的地址。但是对于函数来说:函数名==&函数名
我们代码验证一下(示例):
#include<stdio.h>
int Add(int x,int y)
{
return x+y;
}
int main()
{
printf("%p\n",&Add);
printf("%p\n",Add);
return 0;
}
显然,打印出来的地址是一样的,但是这个时候也会有同学跳出来说:“那数组名和&数组名打印出来的地址还一样呢,但意义明显不一样啊”。但是你想想,函数也没有首元素等其他玩意啊,它就是它本身啊,它也不会出现什么函数首元素啊。
所以再次声明:
在函数指针这一块 函数名==&函数名,它的意义和值,都是一样的
2.创建函数指针
我们知道,数组指针用来存放数组地址,整形指针用来存放整形地址。。。函数指针也不例外,它用来存放函数地址,我们现在定义一个p来存放Add地址,那它的类型怎么创建?我们来看一下具体步骤:
1.p是一个指针对吧,给它一个*是不是必须的 p变成了 * p。为了确保 * 和 p结合(如果没有括号,*或者p有可能会与其他的一些符号结合,具体参见符号优先级)那我在 * p外面加一个括号便于观看也没有问题吧,也就是(*p)
2.那函数总得有参数啊,比如这里是Add(int x,int y)。参数x和y的类型是int
你指针指向的函数是不是要找一下它的参数。所以(*p)(int,int)
3.那函数还有一个性质啊,有没有返回值,要是有的话,类型呢? 这里以Add为例,它是返回int型,所以我们指针也返回int 型 即int(*p)(int,int)
到这里Add函数指针的类型就创建完成啦即为*int(p)(int ,int)
需要注意的是:不同函数的参数类型和返回值类型是不一样的,到时候需要根据不同函数对类型进行转换,这里只是以Add函数为例,其他函数以此类推
ps:一个快速判别类型的方法——去掉变量的名字,剩下的就是类型
代码如下(示例):
int a = 10;//去掉a 类型int
int arr[10] = { 0 };//去掉arr 类型int [10]
int(*parr)[10] = &arr;//去掉parr 类型int(*)[10],数组指针,指向一个10int型元素的数组
int(*pf)(int, int) = &Add;//去掉pf 类型int(*)(int,int)
3.通过函数指针调用函数的两种方法
法一:
我们平时在调用函数的时候,一般就是函数名( ,)然后把参数传进括号即可,那我们现在有函数指针了呀,指针怎么使用?p不是指向了函数Add嘛,我们用*解引用指针,得到的是地址里的东西,也就是说 *p==Add,用 * p(,)来传参也可以实现Add函数的调用。代码如下:
#include<stdio.h>
int Add(int x, int y)
{
return x + y;
}
int main()
{
int ret = Add(2, 3);
printf("%d\n", ret);//ret=5
int(*p)(int, int) = &Add;//p是一个指向函数Add的指针
ret = (*p)(3, 3);//ret=6
//p指向Add,对p解引用就是Add
//简言之:*p=Add
//我们并不总是可以拿到变量,有时是拿到变量的地址
//对应函数指针同样的道理,有时不直接给你函数,给你函数地址,就这样调用
printf("%d\n", ret);
}
法二:
我们在二.1取函数地址那一块介绍了,在函数指针这一块,函数名==&函数名, 也就是说创建函数指针的时候可以这样写:int(*p)(int, int) = Add,Add是赋给了p啊,你也可以认为:p就是Add。你可以这样理解,法一是int(*p)(int, int) = &Add,是把Add的地址给p,所以用p来调用函数要解引用一下,但是法二p就是Add,那不用解引用了,直接调用。代码如下:
#include<stdio.h>
int Add(int x, int y)
{
return x + y;
}
int main()
{
//我们由前面的知识知道:函数add取地址时,add=&add
int(*p)(int, int) = Add;//把Add赋给p,这里p即可看做Add
//与法一不同的是,法一将&Add赋给p,p是Add的地址,所以要解引用,这里p就可以看做是Add本身,可以不解引用
int ret = p(3, 6);
printf("%d", ret);
}//如果是为了方便理解,一般是用第一种方法,如果是为了操作方便,可以用第二种方法
三、函数指针进阶
大家来看这样一个代码( * (void(*)() ) 0)(),乍一看非常复杂,我们来细化一下
1 . ( * (void( * )() ) 0)() 我们抽出加粗部分
这是我们熟悉的老朋友:void( * )(),这不就是一个函数指针嘛,该函数无参,返回类型void
2 . (void( * )() ) 0是什么?我们联想一下(int)3.14,不就是对3.14强制类型转换嘛,将3.14这个浮点型强制转换成整形。这里同样的道理,是将整形0强制转换成类型为void( * )()的一个函数指针
3 .现在有了(void( * )() ) 0,我们在这个东西前面加一个 *,这个是什么意思,我们知道(void( * )() ) 0已经被转换成一个指针(指针即地址)了,地址前面加一个 *表示解引用,取出地址里的东西,也就是找到了那个函数
4 .(void( * )() ) 0表示那个函数那再在后面加一个()即是对函数的调用,也就是( * (void(*)() ) 0)()
总结
提示:本文介绍了函数指针的原理和多种使用方法,对于函数指针想要进阶提升的小伙伴一定要认真研读本文中的进阶题目,指针是一个大头,但相信坚持不懈的你一定可以战胜它,加油!