使用c++的成员指针实现类似Borland VCL组件的事件回调

转载 2012年10月19日 09:18:57

  相信用过Borland delphi或者C++ builder的朋友都应该对VCL组件中的事件回调机制有深刻印象,VCL组件大量的使用了事件属性来简化类之间的交互,提高了VCL组件开发程序的效率。同时,也可以在自己编写的的类中增加事件属性,使与VCL组件或者其他自定义类的交互变得简单、直观。

       VCL的事件机制其实就是函数指针回调的一种形式,通过在一个类A中保存其类B实例方法指针,类A就可以在其内部直接调用类B的实例方法。只是Borland从开发语言层面上把其包装得易于理解和易用。如下面的例子:

       //声明一种事件类型,相当于c++中的函数指针类型,只是“of object”限定了此类型针对的是类方法。

       TErrorNotifyEvent = procedure (ErrCode:integer; ErrMsg:string) of object;

       TSourceClass=class(TObject)      //假设TSourceClass需要把其内部运行的错误通知给其他类的实例

        private

              //我们可以声明一个TErrorNotifyEvent类型的成员变量,用于保存回调函数指针

              FOnError:TErrorNotifyEvent;

        protected

              procedure DoErrorNotify(ErrCode:integer,ErrMsg:string);

        public

              //声明事件属性,并通过FOnError成员变量存取

              property OnError:TErrorNotifyEvent read FOnError write FOnError;

       End;

 

       procedure TSourceClass.DoErrorNotify(ErrCode:integer,ErrMsg:string);

       begin

              if FOnError<>nil then FOnError(ErrorCode,ErrMsg); //在TSrouceClass中回调FOnError保存的方法指针。

       end;

 

       这样,其他类就可以通过存取TSourceClass类的OnError属性达到使用TSourceClass错误报告的目的。一旦       TSourceClass内部有任何的错误需要通知到外部,都可以直接调用DoErrorNotify

 

       TTargetClass=class(TObject) //假设TTargetClass需要TSourceClass的错误通知

         private

             

         public

        //声明一个与TErrorNotifyEvent类型兼容的成员方法

              procedure ReceiveErrorNotify(ErrorCode:integer; ErrMsg:string);

       End;

 

       procedure TTargetClass.ReceiveErrorNotify(ErrorCode:integer; ErrMsg:string);

       begin

              //在ReceiveErrorNotify处理来自TSourceClass错误通知

       end;

       这样,TSourceClass与TTargetClass都已经具备了使用TErrorNotifyEvent事件类型进行交互的一切。下面的            代码演示了如何在它们的实例之间搭起联系。

       objSource:TSourceClass;

       objTarget:TTargetClass;

       objSource:=TSourceClass.Create;

       objTarget:=TTargetClass.Create;

       objSource.OnError:=objTarget.Receive //这样就把objSource与objTarget联系在一起。

 

 

       回到在c++可视化编成中占据重要地位的VC++,其MFC框架就没有提供如VCL框架类似的事件回调机制。不同类之间的交互需要编写很多额外的代码,或者使用其他的方法,如window消息。如使用MFC的CAsyncSocket类时,你不得不通过重载某些方法以达到接收socket数据的目的。如果CAsyncSocket本身有类似socket数据到达的事件通知OnDataArrived,那么我们就可以不需要重载CAsyncSocket类,直接在主程序类中使用OnDataArrived就可以达到接收socket数据的目的。

      

       那么,有没有别的方法可以帮助在VC中实现类似的VCL事件回调机制呢?

 

       参照上面VCL的例子,我们很自然想到形如以下的方式:

typedef  void  (*NOTIFY_EVENT)(int notify_code);  //定义事件回调函数指针类型

 

class A

{

private:

 

public:

   NOTIFY_EVENT OnNotify; //声明事件属性

};

 

class B

{

private:

 

public:

  void  ReceiveNotify(int notify_code) //定义接收回调通知的成员函数

  {

 

  }

};

 

 

       并且按如下方式使用:

       A objA;

       B objB;

       objA.OnNotify=objB.ReceiveNotify;  //搭建类实例的之间联系,但此语句编译出错!

 

 

       在VC中编译,会产生如下的编译错误

              error C2440: '=' : cannot convert from 'void (__thiscall B::*)(int)' to 'void (__cdecl *)(int)'

 

       上述的编译信息表明两点:

       1.类A的OnNOtify成员变量是NOTIFY_EVENT的调用方式与B::ReceiveNotify不同,前者是    __cdecl方      式,后者则是默认的thiscall方式;

       2. NOTIFY_EVENT与B::ReceiveNotify类型不同,前者是一般的函数指针类型,后者则是针对类B  的函数指针类型 。

 

 

成员指针

       顾名思义,就是指向类成员的指针。C++中支持成员指针的定义和使用。如:

       class A

       {

       public:

              int m_IntMember;

              void VoidMethod() {}

       }

       上面的类A中有一个m_IntMember成员变量,一个VoidMethod成员函数。我们可以声明和使用指向它们的成员指针:

       int A::* pInt=&A::m_IntMember;

      

       typedef (B::*METHOD_POINTER)();

       METHOD_POINTER pMethod=&B::VoidMethod;

 

       A objA; A objB;

       int iVar=objA.*pInt; //直接存取实例objA的m_IntMember值

       objA.*pMethod(); //调用的是实例objA的VoidMethod方法

 

       objB.*pInt=iVar; //直接存取实例objB的m_IntMember值

       objB.*pMethod(); //调用的是实例objB的VoidMethod方法

 

 

       那么,我们如何使用成员指针解决上面编译错误的问题呢? 请看下面代码。

  

//声明针对类B的函数指针类型

typedef  void  (B::*ERROR_NOTIFY_EVENT)(int notify_code);

class A

{

private:

 

public:

   ERROR_NOTIFY_EVENT OnNotify;

};

 

class B

{

private:

 

public:

  void  ReceiveNotify(int notify_code)

  {

 

  }

};

 

       A objA;

       B objB;

       objA.OnNotify=objB.ReceiveNotify;  //这样就ok了!!!

 

 

通过声明针对类B的成员函数指针类型 typedef  void  (B::*ERROR_NOTIFY_EVENT)(int notify_code),实现类A实例回调类B实例的目的,这就是c++中实现事件回调机制的方法。


C++中类成员函数作为回调函数 [转]

C++中类成员函数作为回调函数 [转] http://blog.csdn.net/this_capslock/article/details/17001003 回调函数是基于C编程...
  • ksn13
  • ksn13
  • 2014年10月28日 11:42
  • 21191

C++回调函数和this指针的实例

在做到界面开发的过程中,往往需要通过子窗口去操作父窗口的问题。方法很多,大致实现下回调函数和类静态对象的使用。 在类的非静态函数的形参列表里,有个隐含的this指针,但是静态成员函数没有this指针...
  • u010177010
  • u010177010
  • 2016年03月09日 17:26
  • 892

C++中类成员函数作为回调函数

转载自:http://blog.csdn.net/this_capslock/article/details/17001003 回调函数是基于C编程的Windows SDK的技术,不是针对C++的,程...
  • tingzhushaohua
  • tingzhushaohua
  • 2017年08月02日 08:56
  • 239

C++中类成员函数作为回调函数

回调函数是基于C编程的Windows SDK的技术,不是针对C++的,程序员可以将一个C函数直接作为回调函数,但是如果试图直接使用C++的成员函数作为回调函数将发生错误,甚至编译就不能通过。  普通...
  • u012468214
  • u012468214
  • 2013年11月28日 10:45
  • 15434

C++中类成员函数作为回调函数的几种方法

写了一个C++小游戏,想法是间接别人的C语言,但是真正自己操作起来,还是会遇到各种问题,同时也让自己更真切的学到了一些过眼即忘,只有亲自动手才能学到的知识。...
  • lmdyyh5722756
  • lmdyyh5722756
  • 2014年02月23日 22:11
  • 2425

C++类成员函数作为线程回调函数

C++类成员函数作为回调函数的方法
  • a1009563517
  • a1009563517
  • 2016年07月28日 22:47
  • 1519

C/C++之回调函数---函数指针的作用

今天讨论下C/C++中的回调函数。      在理解“回调函数”之前,首先讨论下函数指针的概念。 函数指针 (1)概念:指针是一个变量,是用来指向内存地址的。一个程序运行时,所有和运行相关的物件都是需...
  • kkk0526
  • kkk0526
  • 2013年12月04日 17:24
  • 5367

【C++基础之八】函数指针和回调函数

C++很多类库都喜欢用回调函数,MFC中的定时器,消息机制,hook机制等待,包括现在在研究的cocos2d-x中也有很多的回调函数。 1.回调函数 什么是回调函数呢?回调函数其实就是一个通过函数指针...
  • jackyvincefu
  • jackyvincefu
  • 2013年09月16日 09:40
  • 27956

C++回调机制的几种实现方式

 Callback Callback的本质是设置一个函数指针进去,然后在需要触发某个事件时调用该方法, 比如Windows的窗口消息处理函数就是这种类型。 比如下面的示例代码,我们在...
  • u011555996
  • u011555996
  • 2017年07月26日 13:49
  • 209

让C++回调函数可以同时支持成员函数和静态函数的方法

回调函数是分层软件系统中经常使用的一种技巧,当下层需要调用上层的接口时,一般都使用回调函数来破除循环依赖。在纯C语言环境中,回调函数用起来很简单——一个函数指针而已。但是当大家都用上C++以后,回调函...
  • qq_21895115
  • qq_21895115
  • 2015年05月11日 07:06
  • 884
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章:使用c++的成员指针实现类似Borland VCL组件的事件回调
举报原因:
原因补充:

(最多只允许输入30个字)