详解EventDispatcher事件分发组件


Posted in PHP onDecember 25, 2016

引言

考虑这样一个问题,现在你想给为你的项目提供一个插件系统,插件可以添加一些方法,或者在某些方法执行之前或者之后做些事情,而不干扰其他插件。要实现这个系统,简单的单继承不是个好办法,即使多继承在PHP中是可能的,他也有与生俱来的缺点(多继承不太了解,感觉挺操蛋的)。

Symfony EventDispatcher以一个简单有效的方式实现了中介者模式,事件分发器就是那个中介,让系统和插件不会耦合在一起,这让上面的插件系统成为可能,而且他会让你的项目可扩展性更好。

上面的话,翻译自Symfony官方文档片段

系统剖析

事件存储

详解EventDispatcher事件分发组件

上面这张图是分析Symfony EventDispatcher组件源码得出来的,可以看到事件在系统中是如何存储的

这里面将事件存储了两遍,用来加入优先级priority的概念,存如的时候放入上图中上面的结构中,取出时候从上图中下面的结构中拿出来,相同的事件名称可以有不同的优先级,优先级越高的事件优先触发,优先级相同的时候,先插入的事件优先触发。

排序事件(上图中下面的结构)在插入事件的时候不会构建,而是当取出事件的时候会生成排好序的事件,当相同的事件名中插入新的事件或删除某个事件的时候,会删除对应的排好序的事件名,后面用到的时候重新构建

执行事件的时候,会获取对应事件名排好序的linster列表,按照顺序依次执行。

事件执行

详解EventDispatcher事件分发组件

如上图所示,当触发某个时间的时候,该事件名下面如果监听了多个触发动作,他们会按照优先级、注册顺序依次触发,触发动作一般是一个可执行的“实例”(不管是类还是函数,必须可以通过call_user_func_array调用),可以传入三个参数,第一个参数(必须)是一个Event实例,第二个是触发的事件名,第三个是事件分发器实例。第一个参数会控制事件是否在该事件名下的所有触发动作之间继续传递,比如上面的linstener_2里面将Event.propagationStopped设置为true,执行完linstener_2后,事件就会停止传播,linstener_2后面的动作不会触发。

除此之外,Event实例中还可以保存其他必要的信息,以便linstener触发执行的时候,获取额外的信息。

事件订阅者

详解EventDispatcher事件分发组件

事件订阅者(Event subscriber),告诉dispathcer实例,他要订阅的所有事件,不用一个个通过dispathcer实例去注册。事件订阅者是一个PHP类,他可以告诉dispathcer他要订阅的具体的事件。

好处:

  • 关注的事件不用一个个去注册。
  • 取消关注的事件不用一个个去移除注册。

订阅者内部关注的事件是一个整体,要么全部关注要么全部不关注

实例

普通栗子

include "vendor/autoload.php";
use Symfony\Component\EventDispatcher\EventDispatcher;
use Symfony\Component\EventDispatcher\Event;
class UserEvent extends Event
{
 public function name()
 {
 return "Cartman";
 }

 public function age()
 {
 return "24";
 }
}
$dispatcher = new EventDispatcher();
$dispatcher->addListener("user.name", function($event, $eventName, $dispatcher){
 echo "My name is Cartman\n";
});
$dispatcher->addListener("user.name", function($event, $eventName, $dispatcher){
 echo "My name is {$event->name()} from Event instance\n";
}, 10);
$dispatcher->addListener("user.age", function($event, $eventName, $dispatcher){
 echo "My age is 24\n";
}, 10);
$dispatcher->addListener("user.age", function($event, $eventName, $dispatcher){
 echo "My age is {$event->age()} from Event instance\n";
}, -10);
$dispatcher->dispatch("user.name", new UserEvent());
$dispatcher->dispatch("user.age", new UserEvent());

上面的例子输出

My name is Cartman from Event instance
My name is Cartman
My age is 24
My age is 24 from Event instance

事件订阅者栗子

通过Subscriber注册事件

include "vendor/autoload.php";
use Symfony\Component\EventDispatcher\EventDispatcher;
use Symfony\Component\EventDispatcher\Event;
use Symfony\Component\EventDispatcher\EventSubscriberInterface;
class BookEvent extends Event
{
 public $name = self::class;
}
class BookSubscriber implements EventSubscriberInterface
{
 public static function getSubscribedEvents()
 {
 return [
  "chinese.name" => "chineseNameShow",
  "english.name" => [
  ["englishNameShow", -10],
  ["englishNameAFter", 10],
  ],
  "math.name" => ["mathNameShow", 100]
 ];
 }
 public function chineseNameShow(Event $event)
 {
 echo "我是汉语书籍\n";
 }
 public function englishNameShow(Event $event)
 {
 echo "我是英文书籍\n";
 }
 public function englishNameAFter(Event $event)
 {
 echo "我是展示之后的英文书籍[来自于Event实例{$event->name}]\n";
 }
 public function mathNameShow(Event $event)
 {
 echo "我是展示的数学书籍\n";
 }
}
$dispatcher = new EventDispatcher();
$subscriber = new BookSubscriber();
$dispatcher->addSubscriber($subscriber);
$dispatcher->dispatch("english.name", new BookEvent());
$dispatcher->dispatch("chinese.name");
$dispatcher->removeSubscriber($subscriber);
$dispatcher->dispatch("math.name");

输出为内容:

我是展示之后的英文书籍[来自于Event实例BookEvent]
我是英文书籍
我是汉语书籍

以上就是本文的全部内容,希望本文的内容对大家的学习或者工作能带来一定的帮助,同时也希望多多支持三水点靠木!

PHP 相关文章推荐
php Ajax乱码
Apr 09 PHP
php 动态执行带有参数的类方法
Apr 10 PHP
PHP高级对象构建 工厂模式的使用
Feb 05 PHP
php对二维数组按指定键值key排序示例代码
Nov 26 PHP
php实现mysql数据库分表分段备份
Jun 18 PHP
php+ajax无刷新上传图片实例代码
Nov 17 PHP
PHP生成制作验证码的简单实例
Jun 12 PHP
php模拟post上传图片实现代码
Jun 24 PHP
PHP不使用递归的无限级分类简单实例
Nov 05 PHP
PHP数组内存利用率低和弱类型详细解读
Aug 10 PHP
php对象工厂类完整示例
Aug 09 PHP
php微信公众号开发之简答题
Oct 20 PHP
php 反斜杠处理函数addslashes()和stripslashes()实例详解
Dec 25 #PHP
PHP实现的文件操作类及文件下载功能示例
Dec 24 #PHP
PHP文件与目录操作示例
Dec 24 #PHP
PHP数组操作实例分析【添加,删除,计算,反转,排序,查找等】
Dec 24 #PHP
PHP常见字符串处理函数用法示例【转换,转义,截取,比较,查找,反转,切割】
Dec 24 #PHP
PHP会话控制实例分析
Dec 24 #PHP
PHP面向对象程序设计方法实例详解
Dec 24 #PHP
You might like
一个用于mysql的数据库抽象层函数库
2006/10/09 PHP
PHP数据流应用的一个简单实例
2012/09/14 PHP
jQuery弹出层插件简化版代码下载
2008/10/16 Javascript
javascript笔试题目附答案@20081025_jb51.net
2008/10/26 Javascript
Javascript 函数对象的多重身份
2009/06/28 Javascript
DOM 脚本编程中的兄弟节点
2009/10/31 Javascript
javascript 冒泡排序 正序和倒序实现代码
2010/12/14 Javascript
Jquery 绑定时间实现代码
2011/05/03 Javascript
js异常捕获方法介绍
2013/04/10 Javascript
js模仿windows桌面图标排列算法具体实现(附图)
2013/06/16 Javascript
JS获取下拉列表所选中的TEXT和Value的实现代码
2014/01/11 Javascript
Jquery实现遮罩层的方法
2015/06/08 Javascript
JavaScript中rem布局在react中的应用
2015/12/09 Javascript
超实用的JavaScript表单代码段
2016/02/26 Javascript
javascript中this指向详解
2016/04/23 Javascript
js实现当鼠标移到表格上时显示这一格全部内容的代码
2016/06/12 Javascript
node学习记录之搭建web服务器教程
2017/02/16 Javascript
Angular2开发环境搭建教程之VS Code
2017/12/15 Javascript
vue自定义全局组件(自定义插件)的用法
2018/01/30 Javascript
IE8中jQuery.load()加载页面不显示的原因
2018/11/15 jQuery
jquery检测上传文件大小示例
2020/04/26 jQuery
使用cx_freeze把python打包exe示例
2014/01/24 Python
django 发送手机验证码的示例代码
2018/04/25 Python
Python 查看list中是否含有某元素的方法
2018/06/27 Python
使用OpenCV实现仿射变换—旋转功能
2019/08/29 Python
常用的10个Python实用小技巧
2020/08/10 Python
python 如何把docker-compose.yaml导入到数据库相关条目里
2021/01/15 Python
html5自带表单验证体验优化及提示气泡修改功能
2017/09/12 HTML / CSS
中间件的定义
2016/08/09 面试题
Ruby中的保护方法和私有方法与一般面向对象程序设计语言的一样吗
2013/05/01 面试题
新娘父亲婚礼致辞
2014/01/16 职场文书
机电职业生涯规划书范文
2014/03/08 职场文书
领导班子对照检查剖析材料
2014/10/13 职场文书
2015年话务员工作总结
2015/04/29 职场文书
2016预备党员培训心得体会
2016/01/08 职场文书
python playwrigh框架入门安装使用
2022/07/23 Python