linux C 学习---函数指针
我们经常会听到这样的说法,不懂得函数指针就不是真正的C语言高手。我们不管这句话对与否,但是它都从侧面反应出了函数指针的重要性,所以我们还是有必要掌握对函数指针的使用。先来看看函数指针的定义吧。
函数是由执行语句组成的指令序列或者代码,这些代码的有序集合根据其大小被分配到一定的内存空间中,这一片内存空间的起始地址就成为函数的地址,不同的函数有不同的函数地址,编译器通过函数名来索引函数的入口地址,为了方便操作类型属性相同的函数,c/c++引入了函数指针,函数指针就是指向代码入口地址的指针,是指向函数的指针变量。 因而“函数指针”本身首先应该是指针变量,只不过该指针变量指向函数。这正如用指针变量可指向整形变量、字符型、数组一样,这里是指向函数。C在编译时,每一个函数都有一个入口地址,该入口地址就是函数指针所指向的地址。有了指向函数的指针变量后,可用该指针变量调用函数,就如同用指针变量可引用其他类型变量一样,在这些概念上是一致的。函数指针有两个用途:调用函数和做函数的参数。
函数指针的声明方法为:
数据类型标志符 (指针变量名) (形参列表);
(一)简单的函数指针的应用。
返回类型(*函数名)(参数表)
char (*pFun)(int);
char glFun(int a){ return;}
void main()
{
pFun = glFun;
(*pFun)(2);
}
第一行定义了一个指针变量pFun。首先我们根据前面提到的“形式1”认识到它是一个指向某种函数的指针,这种函数参数是一个int型,返回值是char类型。只有第一句我们还无法使用这个指针,因为我们还未对它进行赋值。
第二行定义了一个函数glFun()。该函数正好是一个以int为参数返回char的函数。我们要从指针的层次上理解函数——函数的函数名实际上就是一个指针,函数名指向该函数的代码在内存中的首地址。
然后就是可爱的main()函数了,它的第一句您应该看得懂了——它将函数glFun的地址赋值给变量pFun。main()函数的第二句中“*pFun”显然是取pFun所指向地址的内容,当然也就是取出了函数glFun()的内容,然后给定参数为2。
下面的程序说明了函数指针调用函数的方法:
#include <stdio.h>
int max ( int x, int y){ return x>y?x:y;}
int min ( int x, int y){ return x<y?x:y;}
void main ()
{
int ( *f ) ( int x, int y)=max;
//f=&max;</span>
printf ( "%d,%d\n", max (2,6), (f)(5,4));
f=min;
printf ("%d,%d\n" , min (2,6), (f)(5,4));
}
注意:以上代码的红色部分我们将会在接下来的代码分析部分进行讲解,读者也可以思考下如果运行注释部分,结果是否还是正确的呢?
f是指向函数的指针变量,所以可把函数max()赋给f作为f的值,即把max()的入口地址赋给f,以后就可以用f来调用该函数,实际上f和max都指向同一个入口地址,不同就是f是一个指针变量,不像函数名称那样是死的,它可以指向任何函数,就看你想怎么做了。在程序中把哪个函数的地址赋给它,它就指向哪个函数。而后用指针变量调用它,因此可以先后指向不同的函数。不过注意,指向函数的指针变量没有++和--运算,用时要小心。
函数括号中的形参可有可无,视情况而定,不过,在某些编译器中这是不能通过的。
执行结果如下:
在上面的描述中留下过一个问题,就是运行注释部分f=&max;结果是否还是正确的呢?下面我就给出上面两个运行结果的对别,然后来分析下原因。
把注释部分加进去的运行结果为:
对比以上的运行结果可以看出,f=&max语句被执行时的结果和没有被执行时的结果是一样的。为什么会出现这样的结果呢?答案是这是编译器处理的,max本身就是个地址,它没有放到任何变量里,自然没有取它的地址一说。
(二)使用typedef更直观更方便
typedef 返回类型(*新类型)(参数表)
typedef char (*PTRFUN)(int);
PTRFUN pFun;
char glFun(int a){ return;}
void main()
{
pFun = glFun;
(*pFun)(2);
}
typedef的功能是定义新的类型。第一句就是定义了一种PTRFUN的类型,并定义这种类型为指向某种函数的指针,这种函数以一个int为参数并返回char类型。后面就可以像使用int,char一样使用PTRFUN了。
第二行的代码便使用这个新类型定义了变量pFun,此时就可以像使用形式1一样使用这个变量了。
#include <stdio.h>
void FileFunc()
{
printf("FileFunc\n");
}
void EditFunc()
{
printf("EditFunc\n");
}
void main()
{
typedef void (*funcp)();
funcp pfun= FileFunc;
pfun();
pfun = EditFunc;
pfun();
}
许多C/C++的面试题都喜欢出一些关于指针的题目,比如:说出下列式子的含义
void * (*(*fp1)(int))[10];
float (*(*fp2)(int, int, float))(int);
typedef double (*(*(*fp3)())[10])();
fp3 a;
int (*(*fp4)[10])();
对于fp1:
我们从里向外一点一点分析,首先(*fp1)(int),这说明fp1是一个函数指针,它有一个int类型的参数;然后我们来找这个函数指针类型的返回值,注意到*(*fp1)(int),所以我们可以断定它的返回值是一个指针,指针指向什么呢?
我们可以看到最外层剩余的部分是void* [10],因此这个函数的返回值是一个指针,这个指针指向一个包含十个void*类型数据的数组。
综上:fp1是一个函数指针,它所指向的函数有一个int类型的参数,并且这个函数的返回值是一个指针,这个指针指向一个包含10个void*元素的数组。
对于fp2
就不再赘述了。fp2是一个函数指针,它所指向的函数有三个参数,参数类型分别为int,int,float;它的返回值是一个函数指针,这个函数指针所指向的函数具有一个int类型的参数,且返回类型为float。
对于fp3
fp3被定义为一个函数指针类型,这种函数指针所指向的函数的参数为空;它的返回值是一个指针,这个指针指向一个包含10元素的函数指针数组,这些函数指针所指向的函数的参数为空,返回值为double。
对于fp4
fp4是一个指针,这个指针指向一个包含10元素的函数指针数组,这些函数指针所指向的函数的参数为空,返回值为int。
《C陷阱与缺陷》中以下面一个例子对函数指针进行了讲解,如下
(*(void(*)())0)();
如果能明白上边几个例子的含义,那么这个简直就是小case啊!
函数指针程序举例
在库函数和系统调用中,有许多函数的原型都设计了函数指针,现在举几个例子,来加深大家对函数指针的理解。
1、线程创建的函数
#include <pthread.h>
int pthread_create(pthread_t *restrict thread,
pthread_attr_t *restrict attr,void *(*start_routine)(void *),void *restrict arg);
该函数的功能就是创建一个线程,第三个参数就是函数指针;
2、信号注册的函数
#include <stdio.h>
typedef void (*sighandler_t)(int);
sighandler_t signal(int sigum,sighandle_t handler);
当然我们也已用方法一来定义
#include <stdio.h>
void (*signal(int sig,void(*func)(int))) (int);
顺便提一下
函数库调用
|
系统调用
|
在所有的ANSI C编译器版本中,C库函数是相同的 | 各个操作系统的系统调用是不同的 |
它调用函数库中的一段程序(或函数) | 它调用系统内核的服务 |
与用户程序相联系 | 是操作系统的一个入口点 |
在用户地址空间执行 | 在内核地址空间执行 |
它的运行时间属于“用户时间” | 它的运行时间属于“系统”时间 |
属于过程调用,调用开销较小 | 需要在用户空间和内核上下文环境间切换,开销较大 |
在C函数库libc中有大约300个函数 | 在UNIX中大约有90个系统调用 |
典型的C函数库调用:system fprintf malloc | 典型的系统调用:chdir fork write brk; |
linux C 学习---函数指针相关推荐
- linux打印函数名,linux kernel 打印函数指针对应的函数名方法
linux kernel 打印函数指针对应的函数名方法 内核中函数指针用的很多,在debug 的时候能直接打印出一个函数指针对应的函数就会很方便. 打印裸指针(raw pointer)用 %p,%p除 ...
- GlusterFS源码学习——函数指针数组的妙用
GlusterFS使用C编写,源码中使用了大量的函数指针来实现模块化编程(是我自己以前没有接触过的编写方式),这样的方式可以很方便的实现功能扩展以及调用. 今天学习的是函数指针数组,也就是把相同系列的 ...
- 通过FFMPEG代码学习函数指针和指针函数
2019独角兽企业重金招聘Python工程师标准>>> 函数指针和指针函数介绍 函数指针和指针函数一直在工作中会用到,现在mark下.部分内容是参考其他人的总结. 1. 函数指针是指 ...
- C语言学习笔记:C语言的指针函数与函数指针??
前言 在C语言里函数.指针这两个词结合的顺序不同其意义也不同,即指针函数与函数指针的意义不同,这是什么情况呢?估计许多学习C语言的小伙伴已经感觉自己懵懵的了,别急且听小编来说个明白. 指针函数 指针函 ...
- C语言学习笔记10-指针(动态内存分配malloc/calloc、realloc、释放free,可变数组实现;Tips:返回指针的函数使用本地变量有风险!;最后:函数指针)
C语言:指针 1. 指针:保存地址的变量 *p (pointer) ,这种变量的值是内存的地址. 取地址符& 只用于获取变量(有地址的东西)的地址:scanf函数-取地址符 地址的大小 ...
- Linux 高并发学习笔记 - Linux 文件操作函数
1.6.2 Linux 文件操作函数 Linux 高并发学习笔记 - 笔记索引 前言 关于文件操作函数这一块主要用英文文档的形势书写,因为凉皮在写文档的时候发现Markdown用起来太繁琐了.那么关于 ...
- C++学习之普通函数指针与成员函数指针
http://blog.csdn.net/lisonglisonglisong/article/details/38353863 函数指针(function pointer)是通过指向函数的指针间接调 ...
- c语言函数指针学习心得,c语言 函数指针 学习C语言笔记
c语言 函数指针 大一学的C语言,课,算是学了一遍,后来接触Java,C++,易语言,python,还有写单片机用的类似C语言,可以说后来根本没有好好学,C也忘个差不多了,但是这次我课选了OC,不喜欢 ...
- C++ 学习笔记(19)new/delete表达式、定位new、typeid、dynamic_cast、type_info、枚举类型、成员函数指针、union、位域、volatile限定符、链接指示
C++ 学习笔记(19)new/delete表达式.定位new.typeid.dynamic_cast.type_info.枚举类型.成员函数指针.union.位域.volatile限定符.链接指示 ...
最新文章
- 机器人售卖雪糕机_智能售货机售货机,冰淇淋厂
- Java中用StreamTokenizer与Scanner读取数据
- Open source robotics toolkits: use virtual arenas to test your robotics algorithms
- 【计算理论】图灵机 ( 接受状态作用 | 格局 | 图灵机语言 | 图灵机设计复杂性 )
- mysql事务 可见性,【每日阅读】2020年12月09日-事务先后的可见性
- SpringSecurity 案例父工程创建
- 提升对前端的认知,不得不了解Web API的DOM和BOM
- duration java_Java Duration类| minusMinutes()方法与示例
- 冈萨雷斯《数字图像处理》读书笔记(十一)——表示和描述
- NTFS系统的访问控制上的权限条目
- 数十篇推荐系统论文被批无法复现:源码、数据集均缺失!
- loadGrid layui
- 大部分程序员还不知道的 Servelt3 异步请求,原来这么简单?
- J2ME Canvas切换闪屏问题解决
- 网络编程-在线英英词典项目
- 9款最佳iPhone WiFi工具和网络分析工具,附下载链接
- MySQL关系一对多一对一多对多
- 浩辰cad2019破解补丁|浩辰cad2019无限试用破解补丁下载(附浩辰cad2019激活工具/无需激活码)
- 写php什么梗,什么是php(php是什么梗)
- 工业机器人编程语言入门_人工智能和机器学习入门的5种编程语言
热门文章
- 答案版2022年3月全国青少年编程考级C语言一级真题
- 普通百姓的创业失败史
- Gradle 使用 Maven 本地仓库。
- 最后,我想对你说一句:我爱你
- 远​程​桌​面​登​陆​后​黑​屏​或​无​法​进​行​操​作​的​解​决​办​法
- Android初级开发(九)——网络交互—WebView的使用
- Linux服务器格式化磁盘,分区,挂载
- 英文求职信计算机网络,英语求职信
- jqGrid学习总结_5 使用formatter
- 软件工程师——操作系统