详解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 中的输出缓冲
Dec 21 PHP
PHP 数组实例说明
Aug 18 PHP
PHP实现定时生成HTML网站首页实例代码
Nov 20 PHP
php生成扇形比例图实例
Nov 06 PHP
Yii框架获取当前controlle和action对应id的方法
Dec 03 PHP
discuz图片顺序混乱解决方案
Jul 29 PHP
使用Huagepage和PGO来提升PHP7的执行性能
Nov 30 PHP
PHP结合Jquery和ajax实现瀑布流特效
Jan 07 PHP
thinkphp3.x中display方法及show方法的用法实例
May 19 PHP
PHP中的浅复制与深复制的实例详解
Oct 26 PHP
PHP中实现中文字串截取无乱码的解决方法
May 29 PHP
PHP常用函数之根据生日计算年龄功能示例
Oct 21 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
PHP 日志缩略名的创建函数代码
2010/05/26 PHP
PHP表单验证的3个函数ISSET()、empty()、is_numeric()的使用方法
2011/08/22 PHP
PHP用身份证号获取星座和生肖的方法
2013/11/07 PHP
PHP中防止SQL注入方法详解
2014/12/25 PHP
高质量PHP代码的50个实用技巧必备(下)
2016/01/22 PHP
php生成毫秒时间戳的实例讲解
2017/09/22 PHP
javascript延时重复执行函数 lLoopRun.js
2007/06/29 Javascript
js实现简单模态窗口,背景灰显
2008/11/14 Javascript
JS input文本框禁用右键和复制粘贴功能的代码
2010/04/15 Javascript
JavaScript 面向对象的之私有成员和公开成员
2010/05/04 Javascript
js选取多个或单个元素的实现代码(用class)
2012/08/22 Javascript
javascript 面向对象封装与继承
2014/11/27 Javascript
JavaScript和HTML DOM的区别与联系及Javascript和DOM的关系
2015/11/15 Javascript
JS实现保留n位小数的四舍五入问题示例
2016/08/03 Javascript
微信小程序 实现tabs选项卡效果实例代码
2016/10/31 Javascript
解决vue项目打包后提示图片文件路径错误的问题
2018/07/04 Javascript
Vue发布项目实例讲解
2019/07/17 Javascript
微信小程序实现点击导航条切换页面
2020/11/19 Javascript
前端 javascript 实现文件下载的示例
2020/11/24 Javascript
JavaScript使用setTimeout实现倒计时效果
2021/02/19 Javascript
vue中axios封装使用的完整教程
2021/03/03 Vue.js
[01:17]辉夜杯战队访谈宣传片—EHOME
2015/12/25 DOTA
python删除本地夹里重复文件的方法
2020/11/19 Python
基于wxPython的GUI实现输入对话框(2)
2019/02/27 Python
python实现两个文件夹的同步
2019/08/29 Python
Python 读取 YUV(NV12) 视频文件实例
2019/12/09 Python
IE10 Error.stack 让脚本调试更加方便快捷
2013/04/22 HTML / CSS
一个SQL面试题
2014/08/21 面试题
什么是静态路由?什么是动态路由?各自的特点是什么?
2015/09/16 面试题
出纳员岗位职责风险
2014/03/06 职场文书
反四风个人对照检查材料
2014/09/26 职场文书
2016特色励志班级口号
2015/12/24 职场文书
七年级数学教学反思
2016/02/17 职场文书
《好妈妈胜过好老师》:每个孩子的优秀都是有源头的
2020/01/03 职场文书
用python自动生成日历
2021/04/24 Python
JS Object构造函数之Object.freeze
2021/04/28 Javascript