回调函数

Posted on 2007-10-02 19:01  hylotta  阅读(425)  评论(0)    收藏  举报

例如在windows消息处理函数,应用程序不直接调用任何窗口函数,而是等待Windows调用窗口函数,请求完成任务或返回信息。为保证Windows调用这个窗口函数,这个函数必须先向Windows登记,然后在Windows实施相应操作时回调,所以窗口函数又称为回调函数。WndProc是一个主回调函数,Windows至少有一个回调函数。

简单说就是,由程序员编写的,给windows系统调用的函数。
也就是说,函数的功能有你定,调用有windows系统调用,不用你关心。你所要做的只是把函数的参数按照规定格式写好,编写好函数。

简单地说:被调用者回头调用调用者的函数(够咬嘴的),故称其为回调找了点回调函数的东西。使用回调函数实际上就是在调用某个函数(通常是API函数)时,将自己的一个函数(这个函数为回调函数)的地址作为参数传递给那个函数。更通行的办法是在函数参数中列一个回调函数地址,并通知调用者:君需自己准备一个比较函数,其中包含两个指针类参数,函数要比较此二指针所指数据之大小,并由函数返回值说明比较结果。回调函数可以象普通函数一样被程序调用,但是只有它被当作参数传递给被调函数时才能称作回调函数。

msdn上这么说的:  
  有关函数指针的知识  
  使用例子可以很好地说明函数指针的用法。首先,看一看   Win32   API   中的   EnumWindows   函数:  
   
  Declare   Function   EnumWindows   lib   "user32"   _  
  (ByVal   lpEnumFunc   as   Long,   _  
  ByVal   lParam   as   Long   )   As   Long  
   
  EnumWindows   是一个枚举函数,它能够列出系统中每一个打开的窗口的句柄。EnumWindows   的工作方式是重复地调用传递给它的第一个参数(lpEnumFunc,函数指针)。每当   EnumWindows   调用函数,EnumWindows   都传递一个打开窗口的句柄。  
   
  在代码中调用   EnumWindows   时,可以将一个自定义函数作为第一个参数传递给它,用来处理一系列的值。例如,可以编写一个函数将所有的值添加到一个列表框中,将   hWnd   值转换为窗口的名字,以及其它任何操作!  
   
  为了表明传递的参数是一个自定义函数,在函数名称的前面要加上   AddressOf   关键字。第二个参数可以是合适的任何值。例如,如果要把   MyProc   作为函数参数,可以按下面的方式调用   EnumWindows:  
   
  x   =   EnumWindows(AddressOf   MyProc,   5)  
   
  在调用过程时指定的自定义函数被称为回调函数。回调函数(通常简称为“回调”)能够对过程提供的数据执行指定的操作。       
  回调函数的参数集必须具有规定的形式,这是由使用回调函数的   API   决定的。关于需要什么参数,如何调用它们,请参阅   API   文档。

我刚开始接触回调时,   也是一团雾水.很多人解释这个问题时,   总是拿API来举例子,   本来菜鸟最惧怕的就是API,   ^_^.   回调跟API没有必然联系.  
   
          其实回调就是一种利用函数指针进行函数调用的过程.  
           
          为什么要用回调呢?比如我要写一个子模块给你用,   来接收远程socket发来的命令.当我接收到命令后,   需要调用你的主模块的函数,   来进行相应的处理.但是我不知道你要用哪个函数来处理这个命令,     我也不知道你的主模块是什么.cpp或者.h,   或者说,   我根本不用关心你在主模块里怎么处理它,   也不应该关心用什么函数处理它......   怎么办?    
   
          使用回调.  
   
          我在我的模块里先定义回调函数类型,   以及回调函数指针.  
          typedef   void   (CALLBACK   *cbkSendCmdToMain)   (AnsiString   sCmd);  
          cbkSendCmdToMain         SendCmdToMain;  
          这样SendCmdToMain就是一个指向拥有一个AnsiString形参,   返回值为void的函数指针.  
   
          这样,   在我接收到命令时,   就可以调用这个函数啦.  
          ...  
          SendCmdToMain(sCommand);  
          ...  
   
          但是这样还不够,   我得给一个接口函数(比如Init),   让你在主模块里调用Init来注册这个回调函数.  
   
          在你的主模块里,   可能这样  
   
          void   CALLBACK   YourSendCmdFun(AnsiString   sCmd);     //声明  
          ...  
          void   CALLBACK   YourSendCmdFun(AnsiString   sCmd);     //定义  
          {  
                  ShowMessage(sCmd);  
          }  
          ...  
   
          调用Init函数向我的模块注册回调.可能这样:  
   
          Init(YourSendCmdFun,   ...);       
          这样,   预期目的就达到了.      
    
          需要注意一点,   回调函数一般都要声明为全局的.   如果要在类里使用回调函数,   前面需要加上   static     ,   其实也相当于全局的.  

回调是一种函数调用机制,   windowsAPI经常使用它,   但是不能说是它的专利.    
  系统调用回调函数,   跟程序员调用回调有什么方式上的区别?  
   
  API使用了回调,   但不是说,   回调只能由API来调用.  

回调函数的作用主要用来在"断点"传输,也就,当调用一个函数,却不能等到函数运行完返回结才去处理别的事情,这样一来,就需要知道,我调用的函数在什么时候结束,这样,我可以进行相应的后续工作,这样,由被调用函数,回调一个在呼叫调用函数中已经定义好的函数(通过指针实现,而且为自动执行的,不用,好像也不能显示调用<因为不知道被调用函数什么时候完成>),这样达到一种程式异步协调的能力.  


回调函数,就是由你自己写的。你需要调用另外一个函数,而这个函数的其中一个参数,就
是你的这个回调函数名。这样,系统在必要的时候,就会调用你写的回调函数,这样你就可
以在回调函数里完成你要做的事。

模块A有一个函数foo,它向模块B传递foo的地址,然后在B里面发生某种事件(event)时,通过从A里面传递过来的foo的地址调用foo,通知A发生了什么事情,让A作出相应反应。 那么我们就把foo称为回调函数。  

例子:

     回调函数是一个很有用,也很重要的概念。当发生某种事件时,系统或其他函数将会自动调用你定义的一段函数。回调函数在windows编程使用的场合很多,比如Hook回调函数:MouseProc,GetMsgProc以及EnumWindows,DrawState的回调函数等等,还有很多系统级的回调过程。本文不准备介绍这些函数和过程,而是谈谈实现自己的回调函数的一些经验。

     之所以产生使用回调函数这个想法,是因为现在使用VC和Delphi混合编程,用VC写的一个DLL程序进行一些时间比较长的异步工作,工作完成之后,需要通知使用DLL的应用程序:某些事件已经完成,请处理事件的后续部分。开始想过使用同步对象,文件影射,消息等实现DLL函数到应用程序的通知,后来突然想到可不可以在应用程序端先写一个函数,等需要处理后续事宜的时候,在DLL里直接调用这个函数即可。  

      于是就动手,写了个回调函数的原形。在VC和 Delphi里都进行了测试

一:声明回调函数类型。

       vc版

              typedef int (WINAPI *PFCALLBACK)(int Param1,int Param2) ;

       Delph版

              PFCALLBACK = function(Param1:integer;Param2:integer):integer;stdcall;

       实际上是声明了一个返回值为int,传入参数为两个int的指向函数的指针。

       由于C++和PASCAL编译器对参数入栈和函数返回的处理有可能不一致,把函数类型用WINAPI(WINAPI宏展开就是__stdcall)或stdcall统一修饰。

二:声明回调函数原形

       声明函数原形

      vc版

          int WINAPI CBFunc(int Param1,int Param2);

       Delphi版

          function CBFunc(Param1,Param2:integer):integer;stdcall;            

      以上函数为全局函数,如果要使用一个类里的函数作为回调函数原形,把该类函数声明为静态函数即可。

三: 回调函数调用调用者

         调用回调函数的函数我把它放到了DLL里,这是一个很简单的VC生成的WIN32 DLL.并使用DEF文件输出其函数名 TestCallBack。实现如下:

              PFCALLBACK  gCallBack=0;
              void WINAPI TestCallBack(PFCALLBACK Func){
                  if(Func==NULL)return;
                  gCallBack=Func;
                  DWORD ThreadID=0;
                  HANDLE hThread = CreateThread(  NULL,  NULL,  Thread1,   LPVOID(0),          &ThreadID );
                  return;
             }

      此函数的工作把传入的 PFCALLBACK Func参数保存起来等待使用,并且启动一个线程。声明了一个函数指针PFCALLBACK gCallBack保存传入的函数地址。

四: 回调函数如何被使用:

          TestCallBack函数被调用后,启动了一个线程,作为演示,线程人为的进行了延时处理,并且把线程运行的过程打印在屏幕上.

本段线程的代码也在DLL工程里实现

      ULONG  WINAPI Thread1(LPVOID Param)
     {
             TCHAR Buffer[256];
             HDC hDC = GetDC(HWND_DESKTOP);
             int Step=1;
             MSG Msg;
              DWORD StartTick;

        //一个延时循环
             for(;Step<200;Step++)
             {
                        StartTick = GetTickCount();
                      /*这一段为线程交出部分运行时间以让系统处理其他事务*/
                       for(;GetTickCount()-StartTick<10;)
                         {
                                 if(PeekMessage(&Msg,NULL,0,0,PM_NOREMOVE) )
                                 {
                                   TranslateMessage(&Msg);
                                   DispatchMessage(&Msg);
                                   }
                           }                                
                           /*把运行情况打印到桌面,这是vcbear调试程序时最喜欢干的事情*/
                           sprintf(Buffer,"Running %04d",Step);
                           if(hDC!=NULL)
                                  TextOut(hDC,30,50,Buffer,strlen(Buffer));
                   }
                   /*延时一段时间后调用回调函数*/  
                   (*gCallback)(Step,1);
                  
                   /*结束*/
                   ::ReleaseDC (HWND_DESKTOP,hDC);
                   return 0
      }

五:万事具备

        使用vc和Delphi各建立了一个工程,编写回调函数的实现部分

       VC版

     int WINAPI CBFunc(int Param1,int Param2)
       {
               int res= Param1+Param2;
             TCHAR Buffer[256]="";
            sprintf(Buffer,"callback result = %d",res);
            MessageBox(NULL,Buffer,"Testing",MB_OK);  //演示回调函数被调用
             return res;            
       }  

         Delphi版
          function CBFunc(Param1,Param2:integer):integer;
          begin
                  result:= Param1+Param2;
                  TForm1.Edit1.Text:=inttostr(result);    / /演示回调函数被调用
           end;   

       使用静态连接的方法连接DLL里的出口函数 TestCallBack,在工程里添加 Button( 对于Delphi的工程,还需要在Form1上放一个Edit控件,默认名为Edit1)。

        响应ButtonClick事件调用 TestCallBack 

              TestCallBack(CBFunc) //函数的参数CBFunc为回调函数的地址 

        函数调用创建线程后立刻返回,应用程序可以同时干别的事情去了。现在可以看到屏幕上不停的显示字符串,表示dll里创建的线程运行正常。一会之后,线程延时部分结束结束,vc的应用程序弹出MessageBox,表示回调函数被调用并显示根据Param1,Param2运算的结果,Delphi的程序edit控件里的文本则被改写成Param1,Param2 的运算结果。

        可见使用回调函数的编程模式,可以根据不同的需求传递不同的回调函数地址,或者定义各种回调函数的原形(同时也需要改变使用回调函数的参数和返回值约定),实现多种回调事件处理,可以使程序的控制灵活多变,也是一种高效率的,清晰的程序模块之间的耦合方式。在一些异步或复杂的程序系统里尤其有用 -- 你可以在一个模块(如DLL)里专心实现模块核心的业务流程和技术功能,外围的扩展的功能只给出一个回调函数的接口,通过调用其他模块传递过来的回调函数地址的方式,将后续处理无缝地交给另一个模块,随它按自定义的方式处理。

      本文的例子使用了在DLL里的多线程延时后调用回调函数的方式,只是为了突出一下回调函数的效果,其实只要是在本进程之内,都可以随你高兴可以把函数地址传递来传递去,当成回调函数使用。

       这样的编程模式原理非常简单单一:就是把函数也看成一个指针一个地址来调用,没有什么别的复杂的东西,仅仅是编程里的一个小技巧。至于回调函数模式究竟能为你带来多少好处,就看你是否使用,如何使用这种编程模式了。

另外的解释:

msdn上这么说的:
有关函数指针的知识
使用例子可以很好地说明函数指针的用法。首先,看一看 Win32 API 中的 EnumWindows 函数:

Declare Function EnumWindows lib "user32" _
(ByVal lpEnumFunc as Long, _
ByVal lParam as Long ) As Long

EnumWindows 是一个枚举函数,它能够列出系统中每一个打开的窗口的句柄。EnumWindows 的工作方式是重复地调用传递给它的第一个参数(lpEnumFunc,函数指针)。每当 EnumWindows 调用函数,EnumWindows 都传递一个打开窗口的句柄。

在代码中调用 EnumWindows 时,可以将一个自定义函数作为第一个参数传递给它,用来处理一系列的值。例如,可以编写一个函数将所有的值添加到一个列表框中,将 hWnd 值转换为窗口的名字,以及其它任何操作!

为了表明传递的参数是一个自定义函数,在函数名称的前面要加上 AddressOf 关键字。第二个参数可以是合适的任何值。例如,如果要把 MyProc 作为函数参数,可以按下面的方式调用 EnumWindows:

x = EnumWindows(AddressOf MyProc, 5)

在调用过程时指定的自定义函数被称为回调函数。回调函数(通常简称为“回调”)能够对过程提供的数据执行指定的操作。

回调函数的参数集必须具有规定的形式,这是由使用回调函数的 API 决定的。关于需要什么参数,如何调用它们,请参阅 API 文档。
回复人:zcchm

我谈一下自己对回调函数的一点理解, 不对的地方请指教.

    我刚开始接触回调时, 也是一团雾水.很多人解释这个问题时, 总是拿API来举例子, 本来菜鸟最惧怕的就是API, ^_^. 回调跟API没有必然联系.

    其实回调就是一种利用函数指针进行函数调用的过程.
   
    为什么要用回调呢?比如我要写一个子模块给你用, 来接收远程socket发来的命令.当我接收到命令后, 需要调用你的主模块的函数, 来进行相应的处理.但是我不知道你要用哪个函数来处理这个命令,  我也不知道你的主模块是什么.cpp或者.h, 或者说, 我根本不用关心你在主模块里怎么处理它, 也不应该关心用什么函数处理它...... 怎么办?

    使用回调.

    我在我的模块里先定义回调函数类型, 以及回调函数指针.
    typedef void (CALLBACK *cbkSendCmdToMain) (AnsiString sCmd);
    cbkSendCmdToMain    SendCmdToMain;
    这样SendCmdToMain就是一个指向拥有一个AnsiString形参, 返回值为void的函数指针.

    这样, 在我接收到命令时, 就可以调用这个函数啦.
    ...
    SendCmdToMain(sCommand);
    ...

    但是这样还不够, 我得给一个接口函数(比如Init), 让你在主模块里调用Init来注册这个回调函数.

    在你的主模块里, 可能这样

    void CALLBACK YourSendCmdFun(AnsiString sCmd);  //声明
    ...
    void CALLBACK YourSendCmdFun(AnsiString sCmd);  //定义
    {
        ShowMessage(sCmd);
    }
    ...

    调用Init函数向我的模块注册回调.可能这样:

    Init(YourSendCmdFun, ...);

    这样, 预期目的就达到了.
    需要注意一点, 回调函数一般都要声明为全局的. 如果要在类里使用回调函数, 前面需要加上 static  , 其实也相当于全局的.

Trackback: http://tb.blog.csdn.net/TrackBack.aspx?PostId=764275

调用(calling)机制从汇编时代起已经大量使用:准备一段现成的代码,调用者可以随时跳转至此段代码的起始地址,执行完后再返回跳转时的后续地址。 CPU为此准备了现成的调用指令,调用时可以压栈保护现场,调用结束后从堆栈中弹出现场地址,以便自动返回。借堆栈保护现场真是一项绝妙的发明,它使调用者和被调者可以互不相识,于是才有了后来的函数和构件,使吾辈编程者如此轻松愉快。若评选对人类影响最大之发明,在火与车轮之后,笔者当推压栈调用。
       话虽这样说,此调用机制并非完美。回调函数就是一例。函数之类本是为调用者准备的美餐,其烹制者应对食客了如指掌,但实情并非如此。例如,写一个快速排序函数供他人调用,其中必包含比较大小。麻烦来了:此时并不知要比较的是何类数据--整数、浮点数、字符串?于是只好为每类数据制作一个不同的排序函数。更通行的办法是在函数参数中列一个回调函数地址,并通知调用者:君需自己准备一个比较函数,其中包含两个指针类参数,函数要比较此二指针所指数据之大小,并由函数返回值说明比较结果。排序函数借此调用者提供的函数来比较大小,借指针传递参数,可以全然不管所比较的数据类型。被调用者回头调用调用者的函数(够咬嘴的),故称其为回调(callback)。
        回调函数使程序结构乱了许多。Windows API 函数集中有不少回调函数,尽管有详尽说明,仍使初学者一头雾水。恐怕这也是无奈之举。无论何种事物,能以树形结构单向描述毕竟让人舒服些。如果某家族中孙
辈又是某祖辈的祖辈,恐怕无人能理清其中的头绪。但数据处理之复杂往往需要构成网状结构,非简单的客户/服务器关系能穷尽。
       Windows 系统还包含着另一种更为广泛的回调机制,即消息机制。消息本是 Windows 的基本控制手段,乍看与函数调用无关,其实是一种变相的函数调用。发送消息的目的是通知收方运行一段预先准备好的代码,相当于调用一个函数。消息所附带的 WParam 和 LParam 相当于函数的参数,只不过比普通参数更通用一些。应用程序可以主动发送消息,更多情况下是坐等 Windows 发送消息。一旦消息进入所属消息队列,便检感兴趣的那些,跳转去执行相应的消息处理代码。操作系统本是为应用程序服务,由应用程序来调用。而应用程序一旦启动,却要反过来等待操作系统的调用。这分明也是一种回调,或者说是一种广义回调。其实,应用程序之间也可以形成这种回调。假如进程B 收到进程 A 发来的消息,启动了一段代码,其中又向进程 A
发送消息,这就形成了回调。这种回调比较隐蔽,弄不好会搞成递归调用,若缺少终止条件,将会循环不已,直至把程序搞垮。若是故意编写成此递归调用,并设好终止条件,倒是很有意思。但这种程序结构太隐蔽,除非十分必要,还是不用为好。
       利用消息也可以构成狭义回调。上面所举排序函数一例,可以把回调函数地址换成窗口handle。如此,当需要比较数据大小时,不是去调用回调函数,而是借 API 函数 SendMessage 向指定窗口发送消息。收到消息方负责比较数据大小,把比较结果通过消息本身的返回值传给消息发送方。所实现的功能与回调函数并无不同。当然,此例中改为消
息纯属画蛇添脚,反倒把程序搞得很慢。但其他情况下并非总是如此,特别是需要异步调用时,发送消息是一种不错的选择。假如回调函数中包含文件处理之类的低速处理,调用方等不得,需要把同步调用改为异步调用,去启动一个单独的线程,然后马上执行后续代码,其余的事让线程慢慢去做。一个替代办法是借 API 函数 PostMessage发送一个异步消息,然后立即执行后续代码。这要比自己搞个线程省事许多,而且更安全。
       如今我们是活在一个 object 时代。只要与编程有关,无论何事都离不开 object。但 object 并未消除回调,反而把它发扬光大,弄得到处都是,只不过大都以事件(event)的身份出现,镶嵌在某个结构之中,显得更正统,更容易被人接受。应用程序要使用某个构件,总要先弄清构件的属性、方法和事件,然后给构件属性赋值,在适当的时候调用适当的构件方法,还要给事件编写处理例程,以备构件代码来调用。何谓事件?它不过是一个指向事件例程的地址,与回调函数地址没什么区别。
不过,此种回调方式比传统回调函数要高明许多。首先,它把让人不太舒服的回调函数变成一种自然而然的处理例程,使编程者顿觉气顺。再者,地址是一个危险的东西,用好了可使程序加速,用不好处处是陷阱,程序随时都会崩溃。现代编程方式总是想法把地址隐藏起来(隐藏比较彻底的如 VB 和 Java),其代价是降低了程序效率。事件例程使编程者无需直接操作地址,但并不会使程序减速。更妙的是,此一改变,本是有损程序结构之奇技怪巧变成一种崭新设计理念,不仅免去被人抨击,而且逼得吾等凡人净手更衣,细细研读,仰慕至今。只是偶然静心思虑,发觉不过一瓶旧酒而已,故引得此番议论,让诸君见笑了。事件驱动程序设计是围绕着消息基础形成的,发生一个事件,伴随着一大堆的消息。
       我理解“回调机制”是window 在执行某个API函数的过程中,调用指定的一个函数。我们可以模拟一下:
假设 ms 提供一个函数叫做  EnumFont ,该函数是得到所有的字体,假设它的实现是
EnumFont()
{
  while ( (f =FindNextFont()) !=NULL)
  {
       printf("fontname: " + f.name);
  }
}
这样就循环显示出所有的字体名称。但是,开发者可能对字体信息另有用处,那么如何才能让开发者能使用这些信息呢,于是做改进:
EnumFont( void*  userFunc )
{
  while ( (f =FindNextFont()) !=NULL)
  {
       printf("fontname: " + f.name);
       if ( userFunc!=NULL)  userFunc( f) ;
  }
}
假设userFunc 是一个函数 void f(
FontObject font).这样使用者只需要定义一个函数:
  void myfunc( FontObject font)
             {     
listCtrl.Addstring ( font.name);            
}
通过使用 EnumFont ( myfunc) 就可以将所有额字体信息添加到一个列表框中。那么我们称 myfunc是一个回调函数,即让某个系统函数调用的函数。因此可以得出结论:
1 回调函数是由开发者按照一定的原型进行定义的函数
2 回调函数并不由发者直接调用执行
3 回调函数通常作为参数传递给系统API,由该API来调用。
4 回调函数可能被系统API调用一次,也可能被循环调用多次。
比如 函数int EnumFontFamilies(
HDC
hdc,             //
handle to device control
  LPCTSTR lpszFamily,  // pointer to family-name string
  FONTENUMPROC lpEnumFontFamProc,
                     
// pointer to callback function
  LPARAM lParam        // pointer to
application-supplied data
);
其中的   FONTENUMPROC lpEnumFontFamProc就是一个回调函数,该函数遵照格式
int CALLBACK EnumFontFamProc( ENUMLOGFONT FAR *lpelf,  NEWTEXTMETRIC FAR
*lpntm, int FontType,  LPARAM lParam )进行定义。
如同mutant所说,回调函数主要用于一些比较费时的操作,或响应不知道何时将会发生的事件,回调函数提供了一种异步的机制,相对于同步执行,提高了效率.前者的例子如WriteFileEx,ReadFileEx等,函数的最后一个参数是一个回调函数的指针,程序中调用WriteFileEx以后,就直接返回了,可以继续进行其他工作,系统在读写操作完成后通知程序作善后处理.后者的例子就是windows的事件机制回调函数的另一个用途,是用于一些枚举函数,如EnumDisplayModes等,每找到一种支持的显示模式,就通知回调函数,由回调函数具体处理,这是因为EnumDisplayModes本身并不知道用户要如何处理.能,用户提供回调函数,定制系统的功能,这样,不同的用户提供不同的回调函数,可以使系统具有不同的功能.这就是所谓的plugin.使用回调函数实际上就是在调用某个函数(通常是API函数)时,将自己的一个函数(这个函数为回调函数)的地址作为参数传递给那个函数。而那个函数在需要的时候,利用传递的地址调用回调函数,这时你可以利用这个机会在回调函数中处理消息或完成一定的操作。至于如何定义回调函数,跟具体使用的API函数有关,一般在帮助中有说明回调函数的参数和返回值等。

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

我们经常在C++设计时通过使用回调函数可以使有些应用(如定时器事件回调处理、用回调函数记录某操作进度等)变得非常方便和符合逻辑,那么它的内在机制如何呢,怎么定义呢?它和其它函数(比如钩子函数)有何不同呢?这里结合自己的使用经历做一个简单的介绍。

使用回调函数实际上就是在调用某个函数(通常是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.h>

int Test1()
{
   int i;
   for (i=0; i<30; i++)
   {
     printf("The %d th charactor is: %c\n", i, (char)('a' + i%26));
 
   }
   return 0;
}
int Test2(int num)
{
   int i;
   for (i=0; i<num; i++)
   {
    printf("The %d th charactor is: %c\n", i, (char)('a' + i%26));
 
   }
   return 0;
}

void Caller1(void (*ptr)())//指向函数的指针作函数参数
{
   (*ptr)();
}
void Caller2(int n, int (*ptr)())//指向函数的指针作函数参数,这里第一个参数是为指向函数的指针服务的,

{                                               //不能写成void Caller2(int (*ptr)(int n)),这样的定义语法错误。
   (*ptr)(n);
   return;
}
int main()
{

   printf("************************\n");
   Caller1(Test1); //相当于调用Test2();
   printf("&&&&&&************************\n");
   Caller2(30, Test2); //相当于调用Test2(30);
   return 0;
}

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

     int (*ptr)(); 这里ptr是一个函数指针,其中(*ptr)的括号不能省略,因为括号的优先级高于星号,那样就成了一个返回类型为整型的函数声明了。


 

博客园  ©  2004-2025
浙公网安备 33010602011771号 浙ICP备2021040463号-3