在KYLib里有大量使用类成员方法指针回调,即所谓的回调事件方法。例如:
class TObject
{
};
/* TKYFmtMemEvent - 格式化内存项事件类 */
class TKYFmtMemEvent
{
public:
// TOnFormat 事件类型
typedef void (TObject::*TDoFormat)(void* AItem, Word ASize);
typedef struct
{
TDoFormat Method;
void* Object;
} TOnFormat;
public:
TKYFmtMemEvent() { Clear(); }
~TKYFmtMemEvent() { Clear(); }
// 清除
void Clear();
// 执行 OnInitialize 事件
void DoInitialize(void* AItem, Word ASize)
{
if (OnInitialize.Method != NULL)
((TObject*)OnInitialize.Object->*OnInitialize.Method)(AItem, ASize);
}
// 执行 OnFinalize 事件
void DoFinalize(void* AItem, Word ASize)
{
if (OnFinalize.Method != NULL)
((TObject*)OnFinalize.Object->*OnFinalize.Method)(AItem, ASize);
}
// 事件
TOnFormat OnInitialize;
TOnFormat OnFinalize;
protected:
private:
};
// 例子:如何设置事件方法指针
void TDemo::SetEvent()
{
FDemo.OnInitialize.Object = this;
FDemo.OnInitialize.Method = (TKYFmtMemEvent::TDoFormat)&TDemo::DoFormat;
// ??? ... ...
}
// FDemo 的 OnInitialize 事件方法
void TDemo::DoFormat(void* AItem, Word ASize)
{
// ??? ... ...
}
回调事件的方法指针需要C++编译器支持,至少VC的不同版本及GCC编译器都支持。
在VC6和VC2003中设置方法指针相对较宽松,VC2005之后就很严格了,如下:
FDemo.OnInitialize.Method = (TKYFmtMemEvent::TDoFormat)&TDemo::DoFormat;
这行代码都被不同版本VC编译器支持,但如下代码就只能被VC6、VC2003支持:
FDemo.OnInitialize.Method = (TKYFmtMemEvent::TDoFormat)DoFormat;
其实,类方法调用原理很简单,若知道如何使用C语言模拟类实现就知道怎么回事了,只是这个工作由编译器来做罢了。不过不是什么方法都可以调用的,如:静态方法就只能当做函数指针来用,而重载方法、虚方法等等是不可靠的,所以最好使用普通的类方法指针。
调用方法与调用函数的区别是在调用方法时,编译器把当前对象的指针当做第一个参数传入,其它参数的传递与函数没有区别,也就说,这为提供回调事件的方法指针提供一条方便之门。
方法指针不能滥用,用好它可以使你的视野更加开阔!