向普通函数传递类成员函数指针的问题

来源:互联网 发布:java curl 参数详解 编辑:程序博客网 时间:2024/04/30 02:56

成员函数func3=>成员函数func2:

将一个类成员函数的函数指针传递给另一个成员函数是比较简单的,只要定义一个函数指针就可以轻松实现。示例如下:

#include <iostream>using namespace std;class test{public: typedef void (test::*pFUN)(); void func1() {  func2(&test::func3); //把func3的指针传递给func2 } void func2(pFUN pfun) {  (this->*pfun)(); } void func3() {  cout<<"test func3."<<endl; }};main(){ void (test::*pfun)() = &test::func3; //把func3的指针赋给pfun test my_test; my_test.func1(); my_test.func2(pfun); //把pfun传递给func2 cin.get();}

输出结果为:test func3.test func3.

但是,如果直接把类成员函数的函数指针作为参数传递给一个外部的普通函数(__cdecl),比如把上面的test类的func3传递给下面的函数:
void fun(void (*pfun)());


编译器就会报错。

原因:

这是因为类的成员函数是绑定到对象的,对于不同的对象,成员函数将会产生不同的拷贝,这样编译器就不知道把哪个拷贝传给fun了。这种情况在很多地方都会遇到,比如OpenCV里面的鼠标事件回调函数:

void cvSetMouseCallback( const char* window_name, CvMouseCallback on_mouse, void* param CV_DEFAULT(NULL));

如果第二个参数是类的成员函数m_class::on_mouse(),那么直接传递就会失败。

解决办法:

解决这个问题的方法,目前我所了解的有四种:

(1)把这个类改成普通的函数。这是最笨的办法。写成类的好处是可以在成员函数之间共享一些数据,如果改成普通的函数,那么就可能会定义一些全局变量来实现共享,而且也不便于管理。

(2)直接把要传递的函数声明为static类型,比如:

static void func3();


这样就把func3变成静态成员,它是绑定在类上面的,对于不同的对象,只有一个拷贝,因此编译器可以确定它的函数指针。但是随之而来的麻烦就是要把它调用到的所有成员(函数和变量)都声明为static类型。这样一来就使所有的对象都可以共享成员变量,使数据变得不安全。另外,如果要传递的函数是重载的虚函数,那么将会报错。因此这种方法具有局限性。
(3)把要传递的函数声明为friend类型,作为友元函数访问类中的成员。友元函数不属于该类,因此可以直接传递给外部的函数,但是同方法(2)一样,需要把所有它调用的成员也声明为static类型,给数据带来不安全。
(4)上面的方法都不太方便。最好的办法是用一个友元(或者全局)函数把要传递的成员函数封装一层,将这个友元函数的函数指针传递给外部的普通函数,而在友元函数内调用该成员函数。这样既不用破坏其他成员函数和变量的属性,保持了类原来的形式,也达到了传递类成员函数指针的目的。实际应用中,回调函数的参数表中一般都含有void *data,这是外部函数留给用户的一个数据接口,因此我们也可以利用这个数据接口将对象的this指针传进友元函数,用this指针调用要传递的类成员函数。

程序如下:

#include <iostream>using namespace std;class test{public:friend void friend_fun(void *obj); // 声明一个友元函数// main中调用外部函数//void call_fun()//{//outside_fun(friend_fun, this); //  ☆ 把友元函数的指针和this指针传递给外部函数 ☆//}private:// 私有成员函数void private_fun(){cout<<"test private_func."<<endl;}};// ☆ 在 友元函数 中封装调用 私有成员函数private_func ☆void friend_fun(void *obj){((test*)obj)->private_fun(); }// ☆ 用友元函数 优于 用static函数 ☆// 外部的普通函数,接受一个函数指针和一个void指针作为参数,比如多线程的执行函数void outside_fun(void (*pfun)(void *data), void *data){(*pfun)(data);}int _tmain(int argc, _TCHAR* argv[]){test* my_test = new test;//my_test->call_fun();outside_fun(friend_fun, (void*)my_test);// 暂停cin.get();return 0;}

输出结果为:test private_func.



原创粉丝点击