详解EventDispatcher事件分发组件

发布时间 - 2026-01-10 22:05:34    点击率:

引言

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

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

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

系统剖析

事件存储

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

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

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

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

事件执行

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

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

事件订阅者

事件订阅者(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]
我是英文书籍
我是汉语书籍

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


# 事件分发  # vue @click @tap重叠事件区分方式  # Vue使用Canvas生成随机大小且不重叠圆  # vue使用swiper.js重叠轮播组建样式  # JS实现三个层重叠点击互相切换的方法  # JavaScript实现多个重叠层点击切换效果的方法  # dispatchEvent解决重叠元素响应事件示例详解  # 我是  # 是一个  # 英文  # 图中  # 第一个  # 他要  # 来自于  # 就会  # 是个  # 也有  # 还可以  # 多个  # 不太  # 而不  # 可以通过  # 第二个  # 会让  # 可以看到  # 或删除  # 所示 


相关栏目: 【 网站优化151355 】 【 网络推广146373 】 【 网络技术251813 】 【 AI营销90571


相关推荐: 如何快速生成橙子建站落地页链接?  BootStrap整体框架之基础布局组件  微信小程序 scroll-view组件实现列表页实例代码  香港服务器选型指南:免备案配置与高效建站方案解析  Laravel如何处理JSON字段_Eloquent原生JSON字段类型操作教程  Python制作简易注册登录系统  如何用wdcp快速搭建高效网站?  javascript和jQuery中的AJAX技术详解【包含AJAX各种跨域技术】  大连网站制作公司哪家好一点,大连买房网站哪个好?  网站设计制作书签怎么做,怎样将网页添加到书签/主页书签/桌面?  历史网站制作软件,华为如何找回被删除的网站?  如何用花生壳三步快速搭建专属网站?  如何快速启动建站代理加盟业务?  重庆市网站制作公司,重庆招聘网站哪个好?  清除minerd进程的简单方法  JavaScript数据类型有哪些_如何准确判断一个变量的类型  阿里云网站搭建费用解析:服务器价格与建站成本优化指南  Laravel如何生成和使用数据填充?(Seeder和Factory示例)  米侠浏览器网页图片不显示怎么办 米侠图片加载修复  C语言设计一个闪闪的圣诞树  Laravel如何实现RSS订阅源功能_Laravel动态生成网站XML格式订阅内容【教程】  1688铺货到淘宝怎么操作 1688一键铺货到自己店铺详细步骤  *服务器网站为何频现安全漏洞?  Laravel如何获取当前登录用户信息_Laravel Auth门面使用与Session用户读取【技巧】  php打包exe后无法访问网络共享_共享权限设置方法【教程】  香港服务器WordPress建站指南:SEO优化与高效部署策略  西安市网站制作公司,哪个相亲网站比较好?西安比较好的相亲网站?  千问怎样用提示词获取健康建议_千问健康类提示词注意事项【指南】  Laravel如何实现事件和监听器?(Event & Listener实战)  Linux系统运维自动化项目教程_Ansible批量管理实战  jquery插件bootstrapValidator表单验证详解  Win11搜索栏无法输入_解决Win11开始菜单搜索没反应问题【技巧】  在线制作视频网站免费,都有哪些好的动漫网站?  如何在万网利用已有域名快速建站?  如何快速生成凡客建站的专业级图册?  VIVO手机上del键无效OnKeyListener不响应的原因及解决方法  Laravel如何发送邮件_Laravel Mailables构建与发送邮件的简明教程  网站广告牌制作方法,街上的广告牌,横幅,用PS还是其他软件做的?  Edge浏览器如何截图和滚动截图_微软Edge网页捕获功能使用教程【技巧】  中山网站推广排名,中山信息港登录入口?  图册素材网站设计制作软件,图册的导出方式有几种?  焦点电影公司作品,电影焦点结局是什么?  laravel怎么在请求结束后执行任务(Terminable Middleware)_laravel Terminable Middleware请求结束任务执行方法  html5如何设置样式_HTML5样式设置方法与CSS应用技巧【教程】  Win11怎么设置虚拟桌面 Win11新建多桌面切换操作【技巧】  Laravel表单请求验证类怎么用_Laravel Form Request分离验证逻辑教程  千库网官网入口推荐 千库网设计创意平台入口  Laravel怎么使用Collection集合方法_Laravel数组操作高级函数pluck与map【手册】  EditPlus中的正则表达式实战(5)  Laravel distinct去重查询_Laravel Eloquent去重方法