C++模拟C#事件委托机制(一)
原文来自于http://www.cnblogs.com/netssfy/articles/1652671.html
写了一段时间的C#代码后确实发现C#的事件委托非常好用。
于是便想是否在C++中也能如此实现。
其实事件委托机制的根本还是回调。由于C#编译器帮我们做了很多事,所以我们在订阅委托时就显的非常的方便。
其实在C++里,实现回调也是非常方便的,函数指针就是为他存在的。
但是,麻烦就麻烦在类成员函数指针的类型上
一个非成员函数void Fun(void)和一个 void Class::Fun(void)的类型是截然不同的。
在C#中,如果定义了一个委托delegate void FunDelegate(void)和事件event FunDelegate event1
那么不管是Class1的成员函数Fun还是Class2上的成员函数Fun都能很方便的订阅到事件上,不需要考虑类的类别
Class1 class1;
Class2 class2;
event1 += class1.Fun;
event1 += class2.Fun;
但是C++则不同,首先Class1中的成员函数Fun的类型为void Class1::Fun(void)
那么这类型的指针就为void Class1::*Fun(void),Class2也同样.
所以他们没有办法定于到一半的函数指针上,即使回返值和参数类型相同,如果这样做,编译器会报类型转换错误.
分析:
对于一个事件委托机制来说,一个事件可以被多个观察者侦听,且不关心这些观察者的类型是什么,即无论是Class1的Fun还是Class2的Fun
都可以侦听同一个类型的事件
设计:
首先,肯定会想到用模板,但是如果CEvent<ObserverType, ArgsType>这样写,那么肯定满足不了上面的条件,因为如果定义
CEvent<typename Class1, typename void>那么肯定不满足,上面的条件,这个时间只能被Class1的实例订阅,那怎么办?
一般来说,根据设计模式,总是存异求同.把相同的部分抽取出来.现在的问题在于CEvent和观察者的类型之间有直接的耦合,要消除这个耦合的话,
我的方法就是加间接.在中间再加一个类型CListenerAgent<typename ObserverType, typename ArgsType>,这个类会知道具体的观察者
类型是什么,并且也知道处理时间的函数的类型(返回值都为void)
然后是消除CEvent对观察者类型的依赖,CEvent只需要触发事件,他不关心到底是哪些类型会去处理这个事件,他要做的只是FireEvent.
也就是说CEvent只需要让CListenerAgent去调度具体是哪个类型的实例会处理这个事件,CEvent只需要使用CListenerAgent,但是我们现在看
到的是CListenerAgent也类型相关的,所以必须抽取出一个接口供CEvent使用:IListenerAgent<typename ArgsType>,这个接口只有一个
函数Fire(ArgsType args).CEvent只要依赖于这个接口就OK了.所以CEvent的定义为CEvent<typename ArgsType>;
下面为源代码
template<typename ArgsType>
class IListenerAgent
{
public:
virtual void Fire(ArgsType) = 0;
};
CListenerAgent<typename ObserverType, typename ArgsType>
{
private:
ObserverType* _instance;
public:
typedef void (ObserverType::*Delegate)(ArgsType);
Delegate Handler;
CListenerAgent()
{
Handler = NULL;
_instance = NULL;
}
CListenerAgent(ObserverType* i, Delegate h)
{
_instance = i;
Handler = h;
}
void Fire(ArgsT ptr)
{
(_instance->*Handler)(ptr);
}
};
template<typename ArgsType>
class CEvent
{
public:
IListenerAgent<ArgsType>* _i;
public:
CEvent()
{
_i = NULL;
}
void operator()(ArgsType args)
{
if (_i = NULL)
{
_i->Fire(args);
}
}
void operator+=(IListenerAgent<ArgsType>* i)
{
_i = i;
}
void operator+=(IListenerAgent<ArgsType>* i)
{
_i = NULL;
}
};
class Observer1
{
public:
void Handler(int)
{
cout << "Observer1::Handler" << endl;
}
};
class Observer2
{
public:
void Handler(int)
{
cout << "Observer2::Handler" << endl;
}
};
这样基本完成了,那么使用起来就像
Observer1 o1;
Observer2 o2;
CListenerAgent<Observer1, void*>* l1 = new CListenerAgent<Observer1, void*>(&o1, &Observer::Handler);
CListenerAgent<Observer2, void*>* l2 = new CListenerAgent<Observer2, void*>(&o2, &Observer::Handler);
CEvent<int> event;
event += l1;
event += l2;
event(10);
到这里,基本的思想就是这么实现.但是这样还是会有很多问题,比如如果事件注册好,但是还没触发时,o1的实例被销毁了,之后事件触发时则会访问
无效的内存地址而引起严重的错误.
在下一篇文章中我会解决这个问题,代码我已经写好了
C++模拟C#事件委托机制(一)的更多相关文章
- C++模拟C#事件委托机制(二)
原文 来自于http://www.cnblogs.com/netssfy/archive/2010/02/02/1662056.html 为了解决非法地址访问的冲突,首先需要知道发生该错误的原因是什么 ...
- Javascript事件模型系列(二)事件的捕获-冒泡机制及事件委托机制
一.事件的捕获与冒泡 由W3C规定的DOM2标准中,一次事件的完整过程包括三步:捕获→执行目标元素的监听函数→冒泡,在捕获和冒泡阶段,会依次检查途径的每个节点,如果该节点注册了相应的监听函数,则执行监 ...
- 【Unity3D技巧】在Unity中使用事件/委托机制(event/delegate)进行GameObject之间的通信 (二) : 引入中间层NotificationCenter
作者:王选易,出处:http://www.cnblogs.com/neverdie/ 欢迎转载,也请保留这段声明.如果你喜欢这篇文章,请点[推荐].谢谢! 一对多的观察者模式机制有什么缺点? 想要查看 ...
- JavaScript 之默认行为 DOM2级,事件委托机制
1. 事件默认行为及阻止方式 1.1 浏览器的默认行为 JavaScript事件本身所具有的属性,例如a标签的跳转,Submit按钮的提交,右键菜单,文本框的输入等. 1.2 ...
- javascript事件委托机制详解
以个人前端工作面试经历来看,javascript事件委托是问的最多的一类题目之一,熟悉事件委托能够了解你对于javascript的掌握程度. 面试官可能问一下问题,现在有5个li待办事件,需要实现当点 ...
- JS 的事件委托机制
以前写上图所示的鼠标点击触发事件,一般都是用如下所示的给每一个表示列表的标签绑定一个click事件(演示用的例子的框架是React): 毫无疑问,这样是比较繁琐的,以后维护修改改个函数名什么的还不方便 ...
- 事件/委托机制(event/delegate)(Unity3D开发之十七)
猴子原创,欢迎转载.转载请注明: 转载自Cocos2Der-CSDN,谢谢! 原文地址: http://blog.csdn.net/cocos2der/article/details/46539433 ...
- js的事件委托机制
如今的JavaScript技术界里最火热的一项技术应该是‘事件委托(event delegation)’了.使用事件委托技术能让你避免对特定的每个节点添加事件监听器:相反,事件监听器是被添加到它们的父 ...
- JS与Jquery的事件委托机制
传送:http://www.ituring.com.cn/article/467 概念: 什么是事件委托:通俗的讲,事件就是onclick,onmouseover,onmouseout,等就是事件,委 ...
随机推荐
- python走起之第九话
协程 协程,又称微线程,纤程.英文名Coroutine.一句话说明什么是线程:协程是一种用户态的轻量级线程. 协程拥有自己的寄存器上下文和栈.协程调度切换时,将寄存器上下文和栈保存到其他地方,在切回来 ...
- eclipse代码自动补全[转]
一.每次输入都自动提示 设置Window->preferences->Java->Editor->Content Assist 再右下角Auto activation trig ...
- c++ STL 学习记录 草稿。
非常丑陋的尝试实现stl.慢慢修改吧. 1)简单实现 vector和list. 2)思索如何开始编写算法. 1,所有容器继承一个抽象容器.那么算法就可以使用抽象基类的next方法来遍历元素. 容器间耦 ...
- Deep Learning 26:读论文“Maxout Networks”——ICML 2013
论文Maxout Networks实际上非常简单,只是发现一种新的激活函数(叫maxout)而已,跟relu有点类似,relu使用的max(x,0)是对每个通道的特征图的每一个单元执行的与0比较最大化 ...
- duilib的caption上的Edit无法激活
当窗口设置标题栏时,鼠标等控件可以相应,edit无法响应. 主要和WindowImplBase::OnNcHitTest 虚函数有关. LRESULT WindowImplBase::OnNcHitT ...
- [阅读笔记]Zhang Y. 3D Information Extraction Based on GPU.2010.
1.立体视觉基础 深度定义为物体间的距离 视差定义为同一点在左图(reference image) 和右图( target image) 中的x坐标差. 根据左图中每个点的视差得到的灰度图称为视差图. ...
- reqwest请求api和约束(转载)
转自:https://www.oschina.net/p/reqwest reqwest 用于浏览器异步HTTP请求.支持xmlHttpRequest, JSONP, CORS, 和 CommonJS ...
- MySQL问题记录--python插入中文至MySQL提示SQLErroor:1366错误
一.在爬虫脚本做以下操作仍提示错误:SQL Error: 1366: Incorrect string value: "\xd0\xc2\xce\xc5-" for column ...
- tensorflow版的bvlc模型
研究相关的图片分类,偶然看到bvlc模型,但是没有tensorflow版本的,所以将caffe版本的改成了tensorflow的: 关于模型这个图: 下面贴出通用模板: from __future__ ...
- PHP控制div块大小和颜色的例子
网站为了设计的更好看,会有很多的样式,而用php来控制样式很常见,无聊写了一个可以用于列表展示的样式,不喜忽喷. 1.先添加一个style样式控制div默认不换行 <style>div{f ...