c++中的回调函数

在C中写回调函数非常方便,C++中定义回调函数要麻烦一些,成员函数牵涉到一个this指针问题。C++中的static成员函数没有this指针,也可以直接作为回调函数使用。C++的成员函数地址是唯一确定的,数据变量地址是可变的,因此编译器使用this指针把这两个部分连接起来,我们调用c++的成员函数,编译器会添加一个this指针到参数中。

有一个类CTest

class CTest
{
public:
void DoMsgFunc1(char* pMsg,int nID)
{
}

void RegiestMsg(int nSrcID,DoMessageFunc pFunc)
{
m_pFunc = pFunc;
}

void HandleMessage(int nMsgID, char* pMsg, int nID)
{
(this->*m_pFunc)(pMsg,nID);
}

private:
DoMessageFunc m_pFunc;
};

定义CTest中的一个回调函数原型:
typedef void (CTest::*DoMessageFunc)(char* pMsg,int nID);
这样在类中就可以直接使用了。

注册回调函数时候,使用如下例子
CTest obj;
obj.RegiestMsg(13,&CTest::DoMsgFunc1);

如果CTest有很多派生类,如
class CTest1 : public CTest

则CTest1注册回调函数的时候,需要强制转换一下,如下
CTest1 obj1;
obj1.RegiestMsg(11,(DoMessageFunc)&CTest1::DoMsgFunc2);

MFC中的消息映射也是同样的道理,能够进行消息映射的类都是从CCmdTarget派生下来的,看看ON_COMMAND宏定义的实现就明白了。

#define ON_COMMAND(id, memberFxn) /
{ WM_COMMAND, CN_COMMAND, (WORD)id, (WORD)id, AfxSigCmd_v, /
static_cast AFX_PMSG (memberFxn) },

其中AFX_PMSG的定义如下:
typedef void (AFX_MSG_CALL CCmdTarget::*AFX_PMSG)(void)

C++中,回调函数可以用多种方式实现,下面分别给出基于函数指针、`std::function` 和 lambda 表达式的回调函数处理示例。 ### 基于函数指针的回调函数示例 ```cpp #include <iostream> // 回调函数类型 typedef void (*Callback)(int); // 接受回调函数作为参数的函数 void processData(int data, Callback callback) { // 模拟处理数据 std::cout << "Processing data: " << data << std::endl; // 调用回调函数 callback(data); } // 回调函数的具体实现 void printData(int data) { std::cout << "Printing data: " << data << std::endl; } int main() { int data = 42; // 调用 processData 函数并传入回调函数 processData(data, printData); return 0; } ``` 在这个示例中,定义了一个函数指针类型 `Callback`,并将其作为参数传递给 `processData` 函数。当 `processData` 函数处理完数据后,调用传入的回调函数 `printData`。 ### 基于 `std::function` 和 `std::bind` 的回调函数示例 ```cpp #include <iostream> #include <functional> // 接受 std::function 作为参数的函数 void processData(int data, std::function<void(int)> callback) { std::cout << "Processing data: " << data << std::endl; callback(data); } // 普通函数 void printData(int data) { std::cout << "Printing data: " << data << std::endl; } class MyClass { public: void memberFunction(int data) { std::cout << "Member function printing data: " << data << std::endl; } }; int main() { int data = 42; // 使用普通函数作为回调 processData(data, printData); MyClass obj; // 使用 std::bind 绑定类的成员函数作为回调 auto boundMemberFunction = std::bind(&MyClass::memberFunction, &obj, std::placeholders::_1); processData(data, boundMemberFunction); return 0; } ``` 在这个示例中,使用 `std::function` 来封装回调函数,它可以接受普通函数、类的成员函数等。使用 `std::bind` 来绑定类的成员函数。 ### 基于 lambda 表达式的回调函数示例 ```cpp #include <iostream> #include <functional> // 接受 std::function 作为参数的函数 void processData(int data, std::function<void(int)> callback) { std::cout << "Processing data: " << data << std::endl; callback(data); } int main() { int data = 42; // 使用 lambda 表达式作为回调 processData(data, [](int data) { std::cout << "Lambda function printing data: " << data << std::endl; }); return 0; } ``` 在这个示例中,使用 lambda 表达式作为回调函数,它可以在调用 `processData` 函数时直接定义回调逻辑。
评论
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值