答案是实现一个基于std::function和模板的C++信号槽机制。通过Signal模板类存储可调用对象,支持普通函数、lambda和成员函数绑定;connect方法注册槽函数,emit触发所有槽;示例展示三种槽的连接与调用,输出对应消息;核心思想为解耦与回调,可扩展disconnect、线程安全及自动解绑等功能。
实现一个简单的C++信号和槽机制,核心是让对象在发出信号时自动调用绑定的函数(槽)。这种机制类似于Qt中的信号槽系统,但我们可以用现代C++(C++11及以上)特性如std::function、std::bind和std::vector来简化实现。
信号(Signal)是一个可被触发的对象,当它被发出时,会调用所有绑定到它的槽函数。槽可以是普通函数、成员函数或lambda表达式。我们需要:
std::function存储任意可调用对象下面是一个支持带一个参数的信号类模板实现:
#include#include #include template class Signal { public: // 使用std::function定义槽类型 using SlotType = std::function ; // 连接一个槽函数 void connect(SlotType slot) { slots.push_back(slot); } // 发出信号,调用所有绑定的槽 void emit(Args... args) { for (auto& slot : slots) { slot(args...); } } private: std::vector slots; };
演示如何连接普通函数、lambda和成员函数作为槽:
// 普通函数作为槽 void globalSlot(int value) { std::cout << "全局函数收到: " << value << "\n"; } struct Receiver { void memberSlot(int value) { std::cout << "成员函数收到: " << value << "\n"; } }; int main() { Signal
sig; // 绑定全局函数 sig.connect(globalSlot); // 绑定lambda sig.connect([](int value) { std::cout << "Lambda收到: " << value << "\n"; }); // 绑定成员函数 Receiver recv; sig.connect([&](int value) { recv.memberSlot(value); }); // 触发信号 sig.emit(42); return 0; }
输出结果:
全局函数收到: 42这个简易版本适合学习和轻量使用。实际中可能需要增强以下功能:
std::weak_ptr配合实现)基本上就这些。这个实现虽然简单,但展示了信号槽的核心思想:解耦发送者和接收者,通过注册回调实现事件驱动编程。不复杂但容易忽略细节,比如内存管理和调用顺序。可以根据项目需求逐步扩展功能。