📄 什么是回调函数(callback)_诸神的黄昏.htm
字号:
#ft A {
COLOR: #7777cc; TEXT-DECORATION: underline
}
#ft A:link {
COLOR: #7777cc; TEXT-DECORATION: underline
}
#ft A:visited {
COLOR: #7777cc; TEXT-DECORATION: underline
}
#usrbar {
LETTER-SPACING: normal
}
#usrbar A {
LETTER-SPACING: normal
}
#usrbar A:link {
LETTER-SPACING: normal
}
#usrbar A:visited {
LETTER-SPACING: normal
}
#ft {
LETTER-SPACING: normal
}
#ft A {
LETTER-SPACING: normal
}
#ft A:link {
LETTER-SPACING: normal
}
#ft A:visited {
LETTER-SPACING: normal
}
</STYLE>
<DIV id=usrbar><NOBR>
<SCRIPT> var myref = encodeURI("http://hi.baidu.com/dreadlord1984/blog/item/19f10ad3bbcb18013bf3cf57%2Ehtml");</SCRIPT>
<A href="http://www.baidu.com/" target=_blank>百度首页</A> | <A
href="http://hi.baidu.com/" target=_blank>百度空间</A>
<SCRIPT language=JavaScript>
document.write(" | <a href='http://passport.baidu.com/?login&tpl=sp&tpl_reg=sp&u="+myref+"'>登录</a>");
</SCRIPT>
</NOBR></DIV>
<DIV id=main align=left><!--[if IE]>
<SCRIPT>
var objmain = document.getElementById("main");
function updatesize(){ var bodyw = window.document.body.offsetWidth; if(bodyw <= 790) objmain.style.width="772px"; else if(bodyw >= 1016) objmain.style.width="996px"; else objmain.style.width="100%"; }
updatesize(); window.onresize = updatesize;
</SCRIPT>
<![endif]-->
<DIV id=header>
<DIV class=lc>
<DIV class=rc></DIV></DIV>
<DIV class=tit><A class=titlink
title="dreadlord1984的空间 http://hi.baidu.com/dreadlord1984"
href="http://hi.baidu.com/dreadlord1984">诸神的黄昏</A></DIV>
<DIV class=desc>知道</DIV>
<DIV id=tabline></DIV>
<DIV id=tab><A href="http://hi.baidu.com/dreadlord1984">主页</A><A class=on
href="http://hi.baidu.com/dreadlord1984/blog">博客</A><A
href="http://hi.baidu.com/dreadlord1984/album">相册</A><SPAN>|</SPAN><A
href="http://hi.baidu.com/dreadlord1984/profile">个人档案</A> <SPAN>|</SPAN><A
href="http://hi.baidu.com/dreadlord1984/friends">好友</A> </DIV></DIV>
<DIV class=stage>
<DIV class=stagepad>
<DIV style="WIDTH: 100%">
<TABLE class=modth cellSpacing=0 cellPadding=0 width="100%" border=0>
<TBODY>
<TR>
<TD class=modtl width=7> </TD>
<TD class=modtc noWrap>
<DIV class=modhead><SPAN class=modtit>查看文章</SPAN></DIV></TD>
<TD class=modtc noWrap align=right></TD>
<TD class=modtr width=7> </TD></TR></TBODY></TABLE>
<DIV class=modbox id=m_blog>
<DIV class=tit>什么是回调函数(callback)</DIV>
<DIV class=date>2007-05-19 23:22</DIV>
<TABLE style="TABLE-LAYOUT: fixed">
<TBODY>
<TR>
<TD>
<DIV class=cnt>
<P>回调函数,就是由你自己写的。你需要调用另外一个函数,而这个函数的其中一个参数,就<BR>是你的这个回调函数名。这样,系统在必要的时候,就会调用你写的回调函数,这样你就可<BR>以在回调函数里完成你要做的事。</P>
<P>模块A有一个函数foo,它向模块B传递foo的地址,然后在B里面发生某种事件(event)时,通过从A里面传递过来的foo的地址调用foo,通知A发生了什么事情,让A作出相应反应。
那么我们就把foo称为回调函数。</P>
<P> </P>
<P>例子:</P>
<P>
回调函数是一个很有用,也很重要的概念。当发生某种事件时,系统或其他函数将会自动调用你定义的一段函数。回调函数在windows编程使用的场合很多,比如Hook回调函数:MouseProc,GetMsgProc以及EnumWindows,DrawState的回调函数等等,还有很多系统级的回调过程。本文不准备介绍这些函数和过程,而是谈谈实现自己的回调函数的一些经验。</P>
<P>
之所以产生使用回调函数这个想法,是因为现在使用VC和Delphi混合编程,用VC写的一个DLL程序进行一些时间比较长的异步工作,工作完成之后,需要通知使用DLL的应用程序:某些事件已经完成,请处理事件的后续部分。开始想过使用同步对象,文件影射,消息等实现DLL函数到应用程序的通知,后来突然想到可不可以在应用程序端先写一个函数,等需要处理后续事宜的时候,在DLL里直接调用这个函数即可。 </P>
<P> 于是就动手,写了个回调函数的原形。在VC和
Delphi里都进行了测试</P>
<P>一:声明回调函数类型。</P>
<P> vc版</P>
<P>
typedef int (WINAPI *PFCALLBACK)(int Param1,int Param2) ;</P>
<P> Delph版</P>
<P>
PFCALLBACK = function(Param1:integer;Param2:integer):integer;stdcall;</P>
<P>
实际上是声明了一个返回值为int,传入参数为两个int的指向函数的指针。</P>
<P>
由于C++和PASCAL编译器对参数入栈和函数返回的处理有可能不一致,把函数类型用WINAPI(WINAPI宏展开就是__stdcall)或stdcall统一修饰。</P>
<P>二:声明回调函数原形</P>
<P> 声明函数原形</P>
<P> vc版</P>
<P>
int WINAPI CBFunc(int Param1,int Param2);</P>
<P> Delphi版</P>
<P> function
CBFunc(Param1,Param2:integer):integer;stdcall; </P>
<P>
以上函数为全局函数,如果要使用一个类里的函数作为回调函数原形,把该类函数声明为静态函数即可。<BR></P>
<P>三: 回调函数调用调用者</P>
<P>
调用回调函数的函数我把它放到了DLL里,这是一个很简单的VC生成的WIN32 DLL.并使用DEF文件输出其函数名
TestCallBack。实现如下:</P>
<P>
PFCALLBACK gCallBack=0;</P>
<P></P>
<P>
void WINAPI TestCallBack(PFCALLBACK Func)</P>
<P>
{</P>
<P>
if(Func==NULL)return;</P>
<P>
gCallBack=Func;</P>
<P>
DWORD ThreadID=0;</P>
<P>
HANDLE hThread = CreateThread( NULL,
NULL, Thread1,
LPVOID(0),
&ThreadID );</P>
<P>
return;</P>
<P>
}</P>
<P> 此函数的工作把传入的 PFCALLBACK
Func参数保存起来等待使用,并且启动一个线程。声明了一个函数指针PFCALLBACK gCallBack保存传入的函数地址。</P>
<P>四: 回调函数如何被使用:</P>
<P>
TestCallBack函数被调用后,启动了一个线程,作为演示,线程人为的进行了延时处理,并且把线程运行的过程打印在屏幕上.</P>
<P>本段线程的代码也在DLL工程里实现</P>
<P> ULONG WINAPI
Thread1(LPVOID Param)</P>
<P> {</P>
<P></P>
<P>
TCHAR Buffer[256];</P>
<P>
HDC hDC = GetDC(HWND_DESKTOP);</P>
<P>
int Step=1;</P>
<P>
MSG Msg;</P>
<P>
DWORD StartTick;</P>
<P> //一个延时循环</P>
<P>
for(;Step<200;Step++)</P>
<P>
{</P>
<P>
StartTick = GetTickCount();</P>
<P>
/*这一段为线程交出部分运行时间以让系统处理其他事务*/</P>
<P>
for(;GetTickCount()-StartTick<10;)</P>
<P>
{</P>
<P>
if(PeekMessage(&Msg,NULL,0,0,PM_NOREMOVE) )</P>
<P>
{</P>
<P>
TranslateMessage(&Msg);</P>
<P>
DispatchMessage(&Msg);</P>
<P>
}</P>
<P>
} </P>
<P>
/*把运行情况打印到桌面,这是vcbear调试程序时最喜欢干的事情*/</P>
<P>
sprintf(Buffer,"Running %04d",Step);</P>
<P>
if(hDC!=NULL)</P>
<P>
TextOut(hDC,30,50,Buffer,strlen(Buffer));</P>
<P>
}</P>
<P></P>
<P>
/*延时一段时间后调用回调函数*/ </P>
<P>
(*gCallback)(Step,1);</P>
<P></P>
<P>
/*结束*/</P>
<P>
::ReleaseDC (HWND_DESKTOP,hDC);</P>
<P>
return 0;</P>
<P> }</P>
<P>五:万事具备</P>
<P>
使用vc和Delphi各建立了一个工程,编写回调函数的实现部分</P>
<P> VC版</P>
<P> int WINAPI CBFunc(int Param1,int
Param2)</P>
<P> {</P>
<P>
int res= Param1+Param2;</P>
<P>
TCHAR Buffer[256]="";</P>
<P>
sprintf(Buffer,"callback result = %d",res);</P>
<P>
MessageBox(NULL,Buffer,"Testing",MB_OK); //演示回调函数被调用</P>
<P>
return
res; </P>
<P> } </P>
<P></P>
<P> Delphi版</P>
<P> function
CBFunc(Param1,Param2:integer):integer;</P>
<P> begin</P>
<P>
result:= Param1+Param2;</P>
<P>
TForm1.Edit1.Text:=inttostr(result); /
⌨️ 快捷键说明
复制代码
Ctrl + C
搜索代码
Ctrl + F
全屏模式
F11
切换主题
Ctrl + Shift + D
显示快捷键
?
增大字号
Ctrl + =
减小字号
Ctrl + -