使用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++中的回调函数   ——指向类成员的指针 在C中我们能够很容易地实现一个指向函数的指针,因此能够方便地实现函数的回调机制。但是在C++中很多人认为类的成员函数不能作为回调函数,因此很多C程序不...
  • doufei_ccst
  • doufei_ccst
  • 2014-03-12 14:44:50
  • 5458

在Borland C++ Builder6中写DLL初步入门

在BORLAND C++ BUILDER6 写DLL初步实验. 虽然简单, 但是基本弄清了点眉目: 主要步骤: 1 写一个DLL   2 写一个APP来调用DLL   1. 写一个DLL FI...
  • gsnet
  • gsnet
  • 2013-02-26 15:58:12
  • 947

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

 进一步通用化       上面定义的ERROR_NOTIFY_EVENT类型只适用于类B,与其他类不兼容,但是其他类有可能需要同样的             错误通知服务,所以有必要使其具备通用性,...
  • lejakwin
  • lejakwin
  • 2004-03-03 11:39:00
  • 497

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

  相信用过Borland delphi或者C++ builder的朋友都应该对VCL组件中的事件回调机制有深刻印象,VCL组件大量的使用了事件属性来简化类之间的交互,提高了VCL组件开发程序的效率。...
  • lejakwin
  • lejakwin
  • 2004-03-03 11:39:00
  • 527

C++学习之对象成员指针篇

所谓对象成员指针就是让一个类的对象的指针作为另外一个类的成员变量,对象成员就是让一个类的对象成为另一个类的成员变量。下面以代码举例说明: 公用坐标类: class Coordinate { p...
  • hudfang
  • hudfang
  • 2016-01-14 18:32:13
  • 1014

<C++略识>之对象指针与对象成员指针

对象指针声明方法: 堆中: Coordinate *p = new Coordinate(); //对象指针p指向了new运算符实例化的对象 栈上: Coordinate a; //从栈上实例...
  • u013003827
  • u013003827
  • 2016-07-21 15:05:09
  • 648

C++的类成员指针的语法

C++的类成员指针的语法
  • aflyeaglenku
  • aflyeaglenku
  • 2017-07-15 15:56:14
  • 475

C++类成员指针的意义

C++中,成员指针是最为复杂的语法结构。但在事件驱动和多线程应用中被广泛用于调用回叫函数。在多线程应用中,每个线程都通过指向成员函数的指针来调用该函数。在这样的应用中,如果不用成员指针,编程是非常困难...
  • captain_wangnb
  • captain_wangnb
  • 2016-01-12 16:54:59
  • 1457

C++的回调机制

什么是回调 开发中经常遇到等待其他模块事件通知的情况,例如: 用户点击UI上button的事件,通知给相关函数处理逻辑Model中数据改变的事件,通知给相关View模块刷新界面异步IO完成的事件...
  • wscdylzjy
  • wscdylzjy
  • 2016-05-16 11:34:10
  • 2719
收藏助手
不良信息举报
您举报文章:使用c++的成员指针实现类似Borland VCL组件的事件回调
举报原因:
原因补充:

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