网站首页 > 技术文章 正文
1.引言:
如果你接触过C#,你就会觉得C#中的delegate(委托)十分灵巧,它的用法上和C\C++的函数指针很像,但是却又比C\C++的函数指针更加灵活。并且委托可以一对多,也就是可以注册多个函数,甚至是某个类的非静态成员函数。而实现事件消息机制【1】也十分依赖于委托机制。基于这样的目的,我们试着在C++上封装出这样的一个委托机制。
【1】值得注意的是这里的委托事件模式与Windows的消息循环体系是不同的,通常Windows的消息是放到消息队列中,应用程序进程从队列中得到消息,然后调用消息处理过程来处理消息,这里是真正的消息通知,并且消息处理过程是有固定的函数声明的,不能更改成其他的格式,但是委托事件模式实际上就是一次函数调用,委托事件模式的使用,其好处是在开发中可以像真正的消息事件体系一样来理解整个体系模式,可以做到很好的接口分离。
2.委托功能使用:
委托使用简单,支持多播,可以添加删除委托。同时支持C++的普通函数、模板函数、类成员函数,类的静态成员函数,并且支持多态。
我们来看一个简单的例子:
e; //将三个函数注册到该委托中 e += newDelegate(NormalFunc); e += newDelegate(A::StaticFunc); e += newDelegate(&A, &A::MemberFunc); //调用 e(1); return 0; }
运行结果:
这里是普通函数 :1
这里是成员静态函数 : 1
这里是成员非静态函数 : 1
由此可以看到将三个函数注册到委托中后,调用委托不仅三个函数不仅能够成功调用,而且参数也是成功传递的。
3.实现无返回值无参数委托的构造
这一部分代码是参照
http://blog.csdn.net/gouki04/article/details/6852394这篇博客上写的。
我们先来看C++中普通函数指针和成员函数指针的区别:
可以看到普通函数指针调用函数的方式和成员非静态函数指针调用函数的方式不同,成员非静态函数指针调用函数需要依赖于该类的一个对象,并且用 .* 或者 ->* 的语法来调用。而成员静态函数调用方式却和普通函数差不多。所以我们需要创建一个委托的基本接口对于不同类型指针的再来派生多态处理。
class IDelegate { public: virtual ~IDelegate { } virtual bool isType(const std::type_info& _type) = 0; virtual void invoke = 0; virtual bool compare(IDelegate *_delegate) const = 0; };这里定义了三个接口,一个是调用,表示调用该Delegate对应的函数指针指向的函数。剩下两个是类型判断,使用了C++的RTTI,动态类型的判断。
接下来我们来派生出能注册普通函数的委托。
class CStaticDelegate : public IDelegate { public: typedef void (*Func); CStaticDelegate(Func _func) : mFunc(_func) { } virtual bool isType(const std::type_info& _type) { return typeid(CStaticDelegate) == _type; } virtual void invoke { mFunc; } virtual bool compare(IDelegate *_delegate) const { if (0 == _delegate || !_delegate->isType(typeid(CStaticDelegate)) ) return false; CStaticDelegate * cast = static_cast
然后是可以注册指向成员非静态函数的指针的委托,因为指向成员非静态函数的类别是这样的 void (ClassName::*FuncName);而ClassName又是不确定的所以我们这里要使用模板类来封装:template
这里的类型T是指这个委托注册的成员函数指针所属的类的类别。比如我注册 A::&MemberFunc ,那么这里的T就被替换为A.其实大家仔细看代码可以发现这两个类十分相似只是invoke 里面调用的方式不同。还有这里的compare判断是指看两个委托指向的成员函数和对象是否一样,如果只是成员函数一样,绑定的对象不一样也视作不同的委托。
这样我们就把C++中的无返回值、无参数的普通函数指针、成员函数指针封装好了。
最后提供统一的接口去生成”函数指针对象“
inline IDelegate* newDelegate( void (*_func) ) { return new CStaticDelegate(_func); } template最后我们我们实现委托,这里我们对多个函数指针的存储使用了STL的list.所以头文件中需要引入
class CMultiDelegate { public: typedef std::list其实最后这个类很像是一个指针容器,然后各个成员方法也只是对这个容器里面的对象进行管理。而主要的三个方法:
重载了 += 表示向这个委托注册一个函数指针,这个方法会自动判重,如果重复了就不会向里面添加。
重载了 -= 表示向这个委托注销一个函数指针,如果这个函数指针不存在就什么也不执行。
重载了 表示当作函数调用启动这个委托,内部就是将所有函数指针指向的函数都运行一遍。
到这里,基本上无返回值、无参数的委托就封装好了。我们先来测试一下:
如果以上代码能够成功运行,那么说明你的第一个版本的委托已经封装完毕,但是如何实现任意返回值、任意参数类型、任意参数个数的函数指针的委托呢?
我在网上查阅过许多代码,发现大多数都是使用的宏替换加上多次引用头文件使得每次编译不同参数个数版本的委托,但是这个方法我感觉巧妙但却鸡肋。后来我尝试着使用C11的新特性:可变模板参数实现了这个需求。能够对用户定义的不同委托去自动生成对应的函数指针类型的委托类。
具体的代码详见 C++实现委托机制(二)
猜你喜欢
- 2025-08-06 中等生如何学好初二数学函数篇
- 2025-08-06 C#构造函数
- 2025-08-06 初中数学:一次函数学习要点和方法
- 2025-08-06 仓颉编程语言基础-数据类型—结构类型
- 2025-08-06 初中VS高中三角函数:从"固定镜头"到"360°全景",数学视野升级
- 2025-08-06 一文讲透PLC中Static和Temp变量的区别
- 2025-08-06 类三剑客:一招修改所有对象!类方法与静态方法的核心区别!
- 2025-05-24 高中数学解题分析方法及知识点
- 2025-05-24 C/C++编程笔记:无法在C++中重载的函数,六种方式
- 2025-05-24 面试与实战:什么是 Lambda?该如何使用?
- 08-06中等生如何学好初二数学函数篇
- 08-06C#构造函数
- 08-06初中数学:一次函数学习要点和方法
- 08-06仓颉编程语言基础-数据类型—结构类型
- 08-06C++实现委托机制
- 08-06初中VS高中三角函数:从"固定镜头"到"360°全景",数学视野升级
- 08-06一文讲透PLC中Static和Temp变量的区别
- 08-06类三剑客:一招修改所有对象!类方法与静态方法的核心区别!
- 1522℃桌面软件开发新体验!用 Blazor Hybrid 打造简洁高效的视频处理工具
- 649℃Dify工具使用全场景:dify-sandbox沙盒的原理(源码篇·第2期)
- 527℃MySQL service启动脚本浅析(r12笔记第59天)
- 492℃服务器异常重启,导致mysql启动失败,问题解决过程记录
- 492℃启用MySQL查询缓存(mysql8.0查询缓存)
- 479℃「赵强老师」MySQL的闪回(赵强iso是哪个大学毕业的)
- 461℃mysql服务怎么启动和关闭?(mysql服务怎么启动和关闭)
- 460℃MySQL server PID file could not be found!失败
- 最近发表
- 标签列表
-
- cmd/c (90)
- c++中::是什么意思 (84)
- 标签用于 (71)
- 主键只能有一个吗 (77)
- c#console.writeline不显示 (95)
- pythoncase语句 (88)
- es6includes (74)
- sqlset (76)
- windowsscripthost (69)
- apt-getinstall-y (100)
- node_modules怎么生成 (87)
- chromepost (71)
- flexdirection (73)
- c++int转char (80)
- mysqlany_value (79)
- static函数和普通函数 (84)
- el-date-picker开始日期早于结束日期 (70)
- asynccallback (71)
- localstorage.removeitem (74)
- vector线程安全吗 (70)
- java (73)
- js数组插入 (83)
- mac安装java (72)
- 查看mysql是否启动 (70)
- 无效的列索引 (74)