C语言中的回调函数实在搞不懂

Python011

C语言中的回调函数实在搞不懂,第1张

typedef struct

{

int a

void (*pshow)(int)

}TMP

void func(TMP *tmp)

{

if(tmp->a >10)//如果a>10,则执行回调函数

{

(tmp->pshow)(tmp->a)

}

}

void show(int a)

{

printf("a的值是%d\n",a)

}

void main()

{

TMP test

test.a = 1

test.pshow = show

func(&test)

}

这只是举例,一般回调函数的用法为:

甲方进行结构体的定义(成员中包括回调函数的指针)

乙方定义结构体变量,并向甲方注册,

甲方收集N个乙方的注册形成结构体链表,在某个特定时刻遍历链表,进行回调。

回调函数,或简称回调,是指通过函数参数传递到其它代码的,某一块可执行代码的引用。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。这一设计允许了底层代码调用在高层定义的子程序。回调的形式因程序设计语言的不同而不同。(C, C++ and Pascal允许将函数指针作为参数传递给其它函数。其它语言,例如JavaScript,Python,Perl和PHP,允许简单的将函数名作为参数传递。Objective-C中允许利用@selector关键字传递SEL类型的函数名。)

回调允许函数调用者在运行时调整原始函数的行为。因为可以把调用者与被调用者分开,所以调用者不关心谁是被调用者。它只需知道存在一个具有特定原型和限制条件的被调用函数。

回调函数使得程序设计更加灵活。

其实,回调函数大多只是自己定义一个名字而已,函数体大多是系统定义好的,它有一个结构,一般一个代回调函数的的函数都有一个参数是接你的回调名的,它把一些值传进回调函数(函数体包括参数是它预定好的,不能自己写,除非全部函数都是你写的),然后回调函数接受值,相应操作后将值返回到原函数体(它的父亲函数),最终让原函数返回一个值

使用回调函数实际上就是在调用某个函数(通常是API函数)时,将自己的一个函数(这个函数为回调函数)的地址作为参数传递给那个函数。而那个函数在需要的时候,利用传递的地址调用回调函数,这时你可以利用这个机会在回调函数中处理消息或完成一定的操作。至于如何定义回调函数,跟具体使用的API函数有关,一般在帮助中有说明回调函数的参数和返回值等。C++中一般要求在回调函数前加CALLBACK(相当于FAR PASCAL),这主要是说明该函数的调用方式。

至于钩子函数,只是回调函数的一个特例。习惯上把与SetWindowsHookEx函数一起使用的回调函数称为钩子函数。也有人把利用VirtualQueryEx安装的函数称为钩子函数,不过这种叫法不太流行。

也可以这样,更容易理解:回调函数就好像是一个中断处理函数,系统在符合你设定的条件时自动调用。为此,你需要做三件事:

1.声明;

2.定义;

3.设置触发条件,就是在你的函数中把你的回调函数名称转化为地址作为一个参数,以便于系统调用。

声明和定义时应注意:回调函数由系统调用,所以可以认为它属于WINDOWS系统,不要把它当作你的某个类的成员函数

回调函数是一个程序员不能显式调用的函数;通过将回调函数的地址传给调用者从而实现调用。回调函数使用是必要的,在我们想通过一个统一接口实现不同的内容,这时用回掉函数非常合适。比如,我们为几个不同的设备分别写了不同的显示函数:void TVshow()void ComputerShow()void NoteBookShow()...等等。这是我们想用一个统一的显示函数,我们这时就可以用回掉函数了。void show(void (*ptr)())使用时根据所传入的参数不同而调用不同的回调函数。

不同的编程语言可能有不同的语法,下面举一个c语言中回调函数的例子,其中一个回调函数不带参数,另一个回调函数带参数。

例子1://Test.c

#include<stdlib.h#include<stdio.hintTest1(){intifor(i=0i<30i++){printf(

The%dthcharactoris:%c

,i,(

char)('a'+i%26))}return0}intTest2(intnum){intifor(i=0i<numi++){printf(

The%dthcharactoris:%c

,i,(

char)('a'+i%26))}return0}voidCaller1(void(*ptr)())//指向函数的指针作函数参数{(*ptr)()}voidCaller2(intn,

int(*ptr)())//指向函数的指针作函数参数,这里第一个参数是为指向函数的指针服务的,//不能写成voidCaller2(int(*ptr)(intn)),这样的定义语法错误。{(*ptr)(n)return}intmain(){printf(

************************)Caller1(Test1)//相当于调用Test2()printf(

&&&&&&************************)Caller2(30

,Test2)//相当于调用Test2(30)return0}

以上通过将回调函数的地址传给调用者从而实现调用,但是需要注意的是带参回调函数的用法。要实现回调,必须首先定义函数指针。函数指针的定义这里稍微提一下。比如: