函数指针用法



1. 函数指针的应用例(1)

#include <stdio.h>
void MyFun1(int x);
void MyFun2(int x);
void MyFun3(int x);
typedef void (*FunType)(int ); // 定义一个函数指针类型FunType,与①函数类型一至
void CallMyFun(FunType fp,int x);

int main(int argc, char* argv[])
{
   CallMyFun(MyFun1,10); //⑤. 通过CallMyFun函数分别调用三个不同的函数
   CallMyFun(MyFun2,20);
   CallMyFun(MyFun3,30);
}
void CallMyFun(FunType fp,int x) //③. 参数fp的类型是FunType。
{
  fp(x);//④. 通过fp的指针执行传递进来的函数,注意fp所指的函数是有一个参数的
}
void MyFun1(int x) // ①. 这是个有一个参数的函数,以下两个函数也相同
{
   printf("函数MyFun1中输出:%d\ni",x);
}
void MyFun2(int x)
{
   printf("函数MyFun2中输出:%d\n",x);
}
void MyFun3(int x)
{
   printf("函数MyFun3中输出:%d\n",x);
}


2. 函数指针应用例(2)

  1. #include<stdio.h>   
  2.   
  3. typedefint(*FP_CALC)(int,int);   
  4. //注意这里不是函数声明而是函数定义,它是一个地址,你可以直接输出add看看  
  5. int add(inta,intb)   
  6. {   
  7.      returna + b;   
  8. }   
  9. int sub(inta,intb)   
  10. {   
  11.      returna - b;   
  12. }   
  13. int mul(inta,intb)   
  14. {   
  15.      returna * b;   
  16. }   
  17. int div(inta,intb)   
  18. {   
  19.      returnb? a/b : -1;   
  20. }   
  21. //定义一个函数,参数为op,返回一个指针。该指针类型为 拥有两个int参数、  
  22. //返回类型为int 的函数指针。它的作用是根据操作符返回相应函数的地址  
  23. FP_CALC calc_func(charop)   
  24. {   
  25.      switch(op)   
  26.       {   
  27.      case'+':returnadd;//返回函数的地址  
  28.      case'-':returnsub;   
  29.      case'*':returnmul;   
  30.      case'/':returndiv;   
  31.      default:   
  32.          returnNULL;   
  33.       }   
  34.      returnNULL;   
  35. }   
  36. //s_calc_func为函数,它的参数是 op,  
  37. //返回值为一个拥有两个int参数、返回类型为int 的函数指针  
  38. int (*s_calc_func(charop)) (int,int)   
  39. {   
  40.      returncalc_func(op);   
  41. }   
  42. //最终用户直接调用的函数,该函数接收两个int整数,和一个算术运算符,返回两数的运算结果  
  43. int calc(inta,intb,charop)   
  44. {   
  45.       FP_CALC fp = calc_func(op); //根据预算符得到各种运算的函数的地址  
  46.          int(*s_fp)(int,int) = s_calc_func(op);//用于测试  
  47.          // ASSERT(fp == s_fp);    // 可以断言这俩是相等的  
  48.      if(fp)returnfp(a, b);//根据上一步得到的函数的地址调用相应函数,并返回结果  
  49.      elsereturn-1;   
  50. }   
  51.   
  52. void main()   
  53. {      
  54.     inta = 100, b = 20;   
  55.   
  56.       printf("calc(%d, %d, %c) = %d\n", a, b,'+', calc(a, b,'+'));   
  57.       printf("calc(%d, %d, %c) = %d\n", a, b,'-', calc(a, b,'-'));   
  58.       printf("calc(%d, %d, %c) = %d\n", a, b,'*', calc(a, b,'*'));   
  59.       printf("calc(%d, %d, %c) = %d\n", a, b,'/', calc(a, b,'/'));   

以上讲的是函数指针,也可以这样用:

#include <stdio.h> 
void say_hello(const char *str) 
{ 
   printf("Hello %s\n", str); 
} 
int main(void) 
{ 
void (*f)(const char *) = say_hello; 
f("Guys"); 
return 0; 
}

除了函数指针,普通函数也可以用typedef,如:


typedef int F (void);//定义函数类型F

之后可以这样声明:

F f,g;相当与声明

int f(void);

int g(void);

下面这个函数声明是错误的:

F h(void);

因为函数可以返回void类型、标量类型、结构体、联合体,但不能返回函数类型,也不能返回数组类型。而下面这个函数声明是正确的:

F *e(void);

函数e返回一个F *类型的函数指针。如果给e多套几层括号仍然表示同样的意思:

F *((e))(void);

但如果把*号也套在括号里就不一样了:

int (*fp)(void);

这样声明了一个函数指针,而不是声明一个函数。fp也可以这样声明:

F *fp;

顺便提下指针函数:

【指针函数】

一个函数不仅可以带回一个整型数据的值,字符类型值和实型类型的值,还可以带回指针类型的数据,使其指向某个地址单元。

        返回指针的函数,一般定义格式为:

        类型标识符    *函数名(参数表)

int *f(x,y);

其中x,y是形式参数,f是函数名,调用后返回一个指向整型数据的地址指针。f(x,y)是函数,其值是指针。

如:char *ch();表示的就是一个返回字符型指针的函数,请看下面的例题:

【例】将字符串1(str1)复制到字符串2(str2),并输出字符串2.

#include "stdio.h"

main()

{

    char *ch(char *,char *);

    char str1[]="I am glad to meet you!";

    char str2[]="Welcom to study C!";

    printf("%s",ch(str1,str2));

}

char *ch(char *str1,char *str2)

{

    int i;

    char *p;

    p=str2
   
    if(*str2==NULL) exit(-1);

    do

    {

        *str2=*str1;

        str1++;

        str2++;

    }while(*str1!=NULL);

    return(p);

}


通过分析可得

函数指针是一个指向函数的指针,而指针函数只是说明他是一个返回值为指针的函数,

函数指针可以用来指向一个函数。

 

回调函数

如果参数是一个函数指针,调用者可以传递一个函数的地址给实现者,让实现者去调用它,这称为回调函数(Callback Function)。例如qsort(3)bsearch(3)

 回调函数示例:void func(void (*f)(void *), void *p);


调用者                                    实现者
  1. 提供一个回调函数,再提供一个准备传给回调函数的参数。

  2. 把回调函数传给参数f,把准备传给回调函数的参数按void *类型传给参数p

  1. 在适当的时候根据调用者传来的函数指针f调用回调函数,将调用者传来的参数p转交给回调函数,即调用f(p);





  • 1
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值