如果试图直接使用C++的成员函数作为回调函数将发生错误,甚至编译就不能通过。通过查询资料发现,其错误是普通的C++成员函数都隐含了一个传递函数作为参数,亦即“this”指针,C++通过传递this指针给其成员函数从而实现程序函数可以访问C++的数据成员。这也可以理解为什么C++类的多个实例可以共享成员函数却-有不同的数据成员。由于this指针的作用,使得将一个CALL-BACK型的成员函数作为回调函数安装时就会因为隐含的this指针使得函数参数个数不匹配,从而导致回调函数安装失败。要解决这一问题的关键就是不让this指针起作用,通过采用以下两种典型技术可以解决在C++中使用回调函数所遇到的问题。这种方法具有通用性,适合于任何C++。
1). 不使用成员函数,为了访问类的成员变量,可以使用友元操作符(friend),在C++中将该函数说明为类的友元即可。 2). 使用静态成员函数,静态成员函数不使用this指针作为隐含参数,这样就可以作为回调函数了。静态成员函数具有两大特点:其一,可以在没有类实例的情况下使用;其二,只能访问静态成员变量和静态成员函数,不能访问非静态成员变量和非静态成员函数。由于在C++中使用类成员函数作为回调函数的目的就是为了访问所有的成员变量和成员函数,如果做不到这一点将不具有实际意义。解决的办法也很简单,就是使用一个静态类指针作为类成员,通过在类创建时初始化该静态指针,如pThis=this,然后在回调函数中通过该静态指针就可以访问所有成员变量和成员函数了。这种处理办法适用于只有一个类实例的情况,因为多个类实例将共享静态类成员和静态成员函数,这就导致静态指针指向最后创建的类实例。为了避免这种情况,可以使用回调函数的一个参数来传递this指针,从而实现数据成员共享。这种方法稍稍麻烦,这里就不再赘述。
举例:
class Test{public:
static void callBackFun(void){}; //因为callBackFun默认有一个const Test* 的指针};
typedef void (*FPtr)(void);
void Fun(FPtr ptr){ ptr();}
void main(void){ Fun(Test::callBackFun); }
#include<cstdlib>
#include<iostream>
#include<cstdio>
#include<cmath>
#include<cstring>
#include<algorithm>
#include<set>
#include<map>
#include<ctime>
#include<list>
#include<queue>
#include<stack>
#include<vector>
#include<fstream>
#define N 1000
using namespace std;
int n,m;
void init()
{
#ifndef ONLINE_JUDGE
freopen("ex.in","r",stdin);
#endif
}
class Test
{
// Test *p;
int a;
public:
static Test *p;
Test(int x=1):a(x){}
static void callBackFun()
{
cout<<"1234"<<endl;
cout<<"##p->a="<<p->a<<endl;
p->a=100;
cout<<"##p->a="<<p->a<<endl;
}
}t;
Test *Test::p=&t;
typedef void (*FPtr)();
void Fun(FPtr ptr)
{
ptr();
}
int main()
{
Fun(Test::callBackFun);
Test t2(1234567);
Test::p=&t2;
Fun(Test::callBackFun);
return 0;
}