0.基本的函数指针的声明与初始化
首先我们来声明一个函数。
int Max_Int(int a, int b)
{
return (a>b)?a:b;
}
把函数名替换成指针表达式是创建指向函数指针最简单的方式:
int (*pFun)(int a, int b); //声明函数指针,此时函数指针没有指向任何函数
这时,我们就已经完成了一个最简单的函数指针声明,其意义是:生命了一个函数指针,其指向的函数返回一个int值,带有两个int型参数。
接下来我们对函数指针进行初始化:
pFun = Max_Int; //使函数指针指向Max_Int函数
这样就完成了函数指针的初始化。接下来,我么就可以通过此函数指针引用此函数。
int max;
int m = 5;
int n = 8;
max = (*pFun)(m,n); //使用函数指针引用Max_Int函数
max = pFun(m,n); //与上一句效果完全相同
printf(“The max value is %d\n”, max); //print ‘8’
PS:第四和第五句效果完全相同,编译器都认识。
1、更高级的声明方式:
1)使用typedef声明函数指针。
我们可以使用typedef来声明函数,以使代码的可读性更强。
typedef bool(*pCopySDMMC2Mem)(int, unsigned int, unsigned short, unsigned int*, bool);
通过之前的介绍,我们可以知道,此函数指针可以指向“返回bool型,带int, unsigned int, unsigned short, unsigned int*, bool型参数” 的函数。
借此声明,我们可以像声明变量那样同时创建多个函数指针。
pCopySDMMC2Mem pf1, pf2;
2)使用define来定义函数指针
#define CopySDMMCtoMem(z,a,b,c,e)(((bool(*)(int, unsigned int, unsigned short, unsigned int*, bool))(*((unsigned int *)0xD0037F98)))(z,a,b,c,e))
效果同上,但指定了指向函数的入口地址(0xD0037F98)
2.函数指针的两个典型应用
1)将函数作为参数传递给函数(此部分转自https://blog.csdn.net/yuexiang321/article/details/52658947)
相信,大家中学的时候都学过积分这个鬼东西。不知道还记不记得积分最原始的计算方法。
对,没错,无限细分,求面积。好的。我们接下来就给大家一个应用函数指针有关积分计算的例子。
直接贴出代码。
#include
//Calculate用于计算积分。一共三个参数。第一个为函数指针func,指向待积分函数。二三参数为积分上下限
double Calculate(double(*func)(double x), double a, double b)
{
double dx = 0.0001;//细分的区间长度
double sum = 0;
for (double xi = a+dx; xi <= b; xi+=dx)
{
double area = func(xi)*dx;
sum +=area;
}
return sum;
}
double func_1(double x)
{
return x*x;
}
double func_2(double x)
{
return x*x*x;
}
void main()
{
printf("%lf\n", Calculate(func_1, 0, 1));
printf("%lf\n", Calculate(func_2, 0, 1));
}
通过函数指针,我们可以在函数中使用别的函数作为参数。此程序可以完成对不同函数的积分。
2)引用不在代码段中的函数
此功能在嵌入式系统中经常使用。我们知道,我们写的用户程序的code是存放在代码段中的,在嵌入式系统中,一般情况下是存放在flash中的。什么叫不在代码段中的函数?很多微控制器在出厂前会将一些功能函数(系统函数)固化在rom中(类似于PC机中的BIOS),如Flash擦写功能,Flash Copy功能。而我们写的代码是不认识这些函数的,不能直接使用函数名调用。所以,当我们想在用户程序中调用这些系统函数时,就只能使用函数指针的方式,通过将系统函数的入口地址传给函数指针,来达到调用rom中程序的目的。这些系统函数一般都会在官方手册中给出功能,返回值类型和参数列表。
下面是从三星的S5PV210_applicationnote中截取的一个系统函数。
从上我们可以分析出,此系统函数的入口地址为0xD0037F98。返回bool型,带有int, unsigned int, unsigned short, unsigned int*, bool型五个参数。实际使用时,我们可以如下调用:
// 实际使用时
pCopySDMMC2Mem p1 = (pCopySDMMC2Mem)0xD0037F98;
p1(x, x, x, x, x);
传入五个合适的参数即可。