指针函数
指针函数: 它的本质是一个函数,不过它的返回值是一个指针。
其声明的形式如下所示:
ret *func(args);
使用举例:
# include <stdio.h>
# include <stdlib.h>
int * func_sum(unsigned int n)
{
static int sum = 0; //注意此次必须为静态,否则函数结束后该变量会被释放
int *p = ∑
for (int i = 0; i < n; i++)
{
sum += i;
}
return p;
}
int main(void)
{
unsigned int num = 0;
printf("please input one number:");
scanf("%d", &num);
int *p = func_sum(num);
printf("sum:%d\n", *p);
return 0;
}
注意:在使用指针函数的时候,一定要避免出现返回局部变量指针的情况。
函数指针
与指针函数不同,函数指针的本质是一个指针,该指针的地址指向了一个函数,所以它是指向函数的指针。
我们知道,函数的定义是存在于代码段,因此,每个函数在代码段中,也有着自己的入口地址,函数指针就是指向代码段中函数入口地址的指针。
声明形式如下所示:
ret (*p)(args);
其中,ret为返回值,*p作为一个整体,代表的是指向该函数的指针,args为形参列表。其中p被称为函数指针变量 。
函数指针的初始化:
与数组类似,在数组中,数组名即代表着该数组的首地址,函数也是一样,函数名即是该数组的入口地址,因此,函数名就是该函数的函数指针。
如下所示:
函数指针变量 = 函数名;
使用举例:
#include <stdio.h>
int max(int a, int b)
{
return a > b ? a : b;
}
int main(void)
{
//函数指针的定义
int (*p)(int, int); //建议
//int (*p)(int a, int b); //建议
//int (*p)(); //不建议
//函数指针初始化
p = max;
//函数指针的调用
int ret = p(10, 15); //建议
//int ret = (*max)(10,15); //不建议
//int ret = (*p)(10,15); //不建议
printf("max = %d \n", ret);
return 0;
}
为什么要使用函数指针?
其实在这样比较简单的代码实现中不容易看出来,当项目比较大,代码变得复杂了以后,函数指针就体现出了其优越性。
举例,如果我们要实现数组的排序,我们知道,常用的数组排序方法有很多种,比如快排,插入排序,冒泡排序,选择排序等,如果不管内部实现,你会发现,除了函数名不一样之外,返回值,包括函数入参都是相同的,这时候如果要调用不同的排序方法,就可以使用指针函数来实现,我们只需要修改函数指针初始化的地方,而不需要去修改每个调用的地方(特别是当调用特别频繁的时候)。
回调函数
函数指针的一个非常典型的应用就是回调函数。
回调函数就是一个通过指针函数调用的函数。其将函数指针作为一个参数,传递给另一个函数。回调函数并不是由实现方直接调用,而是在特定的事件或条件发生时由另外一方来调用的。
使用举例:
#include<stdio.h>
#include<stdlib.h>
//函数功能:实现累加求和
int func_sum(int n)
{
int sum = 0;
if (n < 0)
{
printf("n must be > 0\n");
exit(-1);
}
for (int i = 0; i < n; i++)
{
sum += i;
}
return sum;
}
//这个函数是回调函数,其中第二个参数为一个函数指针
//通过该函数指针来调用求和函数,并把结果返回给主调函数
int callback(int n, int (*p)(int))
{
return p(n);
}
int main(void)
{
int n = 0;
printf("please input number:");
scanf("%d", &n);
//此处直接调用回调函数,而不是直接调用func_sum函数
printf("the sum from 0 to %d is %d\n", n, callback(n, func_sum));
return 0;
}
回调函数callback无需关心func_sum是怎么实现的,只需要去调用即可。
这样的好处就是,如果以后对求和函数有优化,新写了个func_sum2函数的实现,我们只需在调用回调函数的地方将函数指针指向func_sum2即可,而无需去修改callback函数内部。