C语言 函数指针做函数参数(即回调函数)

简介: C语言 函数指针做函数参数(即回调函数)

函数指针做函数参数(回调函数)


函数参数除了是普通变量,还可以是函数指针变量,也就是可以实现回调函数。


所谓回调,就是模块A要通过模块B的某个函数b()完成一定的功能,但是函数b()自己无法实现全部功能,需要反过头来调用模块A中的某个函数a()来完成,这个a()就是回调函数。


回调函数概念


回调函数是程序中一种常用的函数调用方式。其特点是可以在下层软件模块(宏观上的被调用者)中,调用来自上层的软件模块(宏观上的调用者)中的函数,这和通常的调用顺序是相反的。


程序的调用方式软件模块的关系,从总体上看,有上层软件模块和下层软件模块之分。


一般有三种调用方式


  1. 简单的同步调用上调下直接返回


  1. 带有回调的调用双向调用


  1. 异步调用一般基于消息或事件类似MFC的机制


带有回调的调用方式是一种双向调用模式,在这种方式中,下层模块的函数在被调用的时候,将会调用上层模块的某个函数,回调上层模块的函数至少有两个作用,一是监视下层模块函数的运行状态,二是干预下层模块函数的运行。


在C语言中回调函数的基础是函数类型的指针。


回调函数的作用


回调函数的作用:回调函数的表现形式是一个某种类型的函数指针,这个函数由上层的软件模块实现,将这个函数指针通过某种方式传递给下层的软件模块,由下层的软件模块在某个时刻调用这个函数。


函数指针的本质是一个函地址,在32位的系统中,本质是一个32位无符号整数。


函数指针由上层模块传递到下层模块的传递方式是可以通过函数的参数传递,也可以使用结构体的成员进行传递。


回调函数调用时刻


调用回调函数的时刻一般有两种:


  • 一种是在上层模块调用下层模块的函数中,直接调用回调函数:


  • 另一种是使用注册的方式,在某个事件发生的时候下层模块调用回调函数。


回调函数的语法:


1.简单的函数类型为:无参数、无返回值的函数。


简单的回调函数的格式是这种形式。其类型的定义如下所示:


typedef void (*T_SIMPLE_CALLBACK)(void);


注:回调函数可以是任意类型的函数指针,简单的形式是无任何参数,也没有返回值得函数指针


2.完全形式的回调函数


完全形式的回调函数可以按照如下方式实现,它包含两个无符号整数的参数,返回值也是无符号的整数。其类型的定义如下所示:


typedef unsigned int(*T_FULL_CALLBACK)(unsigned int para1,unsigned int para2);


具体例如这样:


//形参为普通变量
void fun( int x ){}
//形参为函数指针变量
void fun( int(*p)(int a) ){}


函数指针变量常见的用途之一:是把指针作为参数传递到其他函数,指向函数的指针也可以作为参数,以实现函数地址的传递。


注:


1.典型的回调函数包含两个整数型参数和整数返回值,第一个参数由下层定义,第二个参数由上层定义,返回值是上层经过回调反馈给下层的


2.事实上,这种回调机制在一个系统的各个层次软件协同处理的时候很常用。例如,下层模块表示一个GUI系统,具有绘制界面的功能,上层将模块将回调函数的指针传递给下层,当界面产生“事件”的时候,调用回调函数执行上层软件模块所定制的行为。


代码案例:模拟计算器


#include<stdio.h>
#include<stdlib.h>
#include<string.h>
int plus(int a, int b)
{
  return a + b;
}
int sub(int a, int b)
{
  return a - b;
}
int mul(int a, int b)
{
  return a * b;
}
int division(int a, int b)
{
  return a / b;
}
//函数指针 做函数的参数 --- 回调函数
void Calculator(int(*myCalculate)(int, int), int a, int b)
{
  int ret = myCalculate(a, b); //dowork中不确定用户选择的内容,由后期来指定运算规则
  printf("ret = %d\n", ret);
}
void test01()
{
  printf("请输入操作符\n");
  printf("1、+ \n");
  printf("2、- \n");
  printf("3、* \n");
  printf("4、/ \n");
  int select = -1;
  scanf("%d", &select);
  int num1 = 0;
  printf("请输入第一个操作数:\n");
  scanf("%d", &num1);
  int num2 = 0;
  printf("请输入第二个操作数:\n");
  scanf("%d", &num2);
  switch (select)
  {
  case  1:
    Calculator(plus, num1, num2);
    break;
  case  2:
    Calculator(sub, num1, num2);
    break;
  case 3:
    Calculator(mul, num1, num2);
    break;
  case 4:
    Calculator(division, num1, num2);
    break;
  default:
    break;
  }
}
int main()
{
  while (1)
  {
    test01();
  }
  return 0;
}


运行结果为:


相关文章
|
2天前
|
安全 编译器 程序员
C语言之字符函数&字符串函数篇(1)
C语言之字符函数&字符串函数篇(1)
23 0
|
2天前
|
C语言 C++
C语言之指针进阶篇_回调函数(3)
C语言之指针进阶篇_回调函数(3)
9 0
|
2天前
|
存储 安全 编译器
C语言之指针篇
C语言之指针篇
34 0
|
2天前
|
算法 C语言
C语言之函数递归篇
C语言之函数递归篇
21 0
|
2天前
|
程序员 编译器 C语言
C语言之函数篇
C语言之函数篇
41 1
|
2天前
|
C语言
C语言 -- 如何传递数组参数
C语言 -- 如何传递数组参数
10 3
|
5天前
|
C语言
C语言变量和指针
C语言变量和指针
5 0
|
5天前
|
C语言 C++
C语言变量、地址、字符及printf()/sizeof()/scanf()函数介绍
C语言变量、地址、字符及printf()/sizeof()/scanf()函数介绍
4 0
|
5天前
|
C语言
C语言数组和函数
C语言数组和函数
7 0
|
5天前
|
C语言
C语言指针使用及动态分配内存
C语言指针使用及动态分配内存
9 0