Symfony2 EventDispatcher组件

一个插件系统中,A插件在不影响其它插件的前提下,添加新的方法,或者在一个方法运行前做一些准备工作,通过继承来实现扩展是很不容易的,由于插件之间的关联关系,A插件的改变也会使得关联的插件被动的修改。

Symfony2的EventDispatcher组件实现了中介者(mediator)模式,实现了插件之间的解耦和关联的关系。

举个栗子,在HttpKernel组件中,一旦Response被创建了,在Response发送给客户端的时候,允许系统的其它模块修改它是很有必要和很有用的(例如:在头部添加缓存字段)。Symfony2内核调度kernel.response事件,那么监听该事件的监听器就可以修改Response对象了:

a) 监听器(PHP对象)告诉一个特定的dispatcher类,它要监听kernel.response事件;

b) 在某一时刻,Symfony就会告诉dispatcher调度kernel.response事件,把一个包含Response对象的Event对象传入事件调度方法EventDispatcher::dispatch()

c) dispatcher通知所有监听kernel.response事件的监听器运行,允许这些监听器修改Response对象。

使用方法

Events

当一个事件被调度,该事件会有一个唯一的名称(例如:kernel.response),可以被任意多个监听器监听。同时实例化一个Event对象,并传入到所有的监听器中。在后面会看到,Event对象包含了被调度事件的数据。

命名规范

事件名可以是任意的字符串,但是可以遵循以下几点命名规范:

* 只使用小写字符、数字、点、下划线;

* 前缀名使用带有点的命名;

* 结束部分的名字使用一个能表达当前事件行为的动词;

事件名和事件对象

dispatcher调度事件通知监听器时候,会把一个Event对象作为参数传入到监听器中。基类的Event非常简单,只有一个停止事件传递给下一个监听器的方法,没有太多别的了。

通常,一个特定事件的数据都会保存到Event对象中,方便把数据传递给监听器。在kernel.response事件中,传入到监听器的Event对象是子类FliterResponseEvent的对象,FliterResponseEvent是Event的子类。FliterResponseEvent包含getReponse和setResponse方法,允许监听器获得或者修改Response对象。

总的来说:创建一个监听特定事件的监听器的时候,Event的一个子类会传入到监听器中,监听器可以通过该子类的方法获得和修改信息。

事件调度器(dispatcher)

dispatcher是事件调度系统的核心,一般来说,一个dispatcher内包含一个监听器的列表,当一个事件需要被调度的时候,dispatcher就会通知它包含的所有监听该事件的监听器。

1 use Symfony\Component\EventDispatcher\EventDispatcher;
2
3 $dispatcher = new EventDispatcher();

关联监听器

把监听器添加到dispatcher上,监听特定的事件,那么该事件被调度的时候,dispatcher就会通知监听器工作了。dispatcher使用addListener方法把一个监听器(PHP callable)添加到一个事件上。

1 $listener = new AcmeListener();
2 $dispatcher->addListener(‘foo.action‘, array($listener, ‘onFooAction‘));

addListener方法有三个参数:

* 监听器需要监听是的事件的名称;

* 监听器(一个PHP callable);

* 一个可选的表示优先级的整数(数值越高优先级越高,监听器就会更早的被触发),默认为0,如果优先级一样,那么谁先添加就先触发;

PHP callable是指能作为参数传入call_user_func()或者传入is_callable()函数执行后返回true的PHP 变量。PHP callable可以是 \Closure实例,一个实现了__invoke方法的对象,或者是表示一个函数的字符串,或者一个表示对象方法或者类方法的数组。

到目前为止,我们看过把一个PHP对象作为监听器,我们也可以把Closure对象作为监听器。

1 use Symfony\Component\EventDispatcher\Event;
2
3 $dispatcher->addListener(‘foo.action‘, function (Event $event) {
4     // will be executed when the foo.action event is dispatched
5 });

    在上面的例子中,foo.action事件被调度,dispatcher就调用AcmeListener::onFooAction方法,并把Event对象作为唯一的参数传入方法中。

 1 use Symfony\Component\EventDispatcher\Event;
 2
 3 class AcmeListener
 4 {
 5     // ...
 6
 7     public function onFooAction(Event $event)
 8     {
 9         // ... do something
10     }
11 }

     在实际使用中,都是传入一个特定的Event子类的对象到监听器,例如FilterResponseEvent:

1 use Symfony\Component\HttpKernel\Event\FilterResponseEvent;
2
3 public function onKernelResponse(FilterResponseEvent $event)
4 {
5     $response = $event->getResponse();
6     $request = $event->getRequest();
7
8     // ...
9 }

创建和调度事件

除了系统内置的事件,我们也可以创建和调度自定义的事件。这是很有好处的,当我们使用第三方类库的时,还有可以使不同的组件之间解耦,使系统更灵活健壮。

静态的Events类

假如我们要创建一个事件——store.order——当订单被创建的时候就会被触发。

namespace Acme\StoreBundle;

final class StoreEvents
{
    /**
     * The store.order event is thrown each time an order is created
     * in the system.
     *
     * The event listener receives an
     * Acme\StoreBundle\Event\FilterOrderEvent instance.
     *
     * @var string
     */
    const STORE_ORDER = ‘store.order‘;
}

这个类并没有什么方法,也不做什么操作,只是定义了事件名称,方便管理和组织事件。监听这个事件的监听器都会被传入一个FilterOrderEvent对象。

创建一个Event对象

接着,当你调度这个新的事件的时候,会创建一个Event对象传如到dispatcher的dispatch()方法,dispatcher就把这个Event对象传给所有的监听该事件的监听器。如果我们不需要向监听器传入任何信息,那么可以使用系统默认的Symfony\Component\EventDispatcher\Event 类。然而,很多时候,我们都需要传入特定的信息到监听器,那么我们可以创建一个类继承Symfony\Component\EventDispatcher\Event。

例如,我们需要在所有的监听器中传入order对象:

 1 namespace Acme\StoreBundle\Event;
 2
 3 use Symfony\Component\EventDispatcher\Event;
 4 use Acme\StoreBundle\Order;
 5
 6 class FilterOrderEvent extends Event
 7 {
 8     protected $order;
 9
10     public function __construct(Order $order)
11     {
12         $this->order = $order;
13     }
14
15     public function getOrder()
16     {
17         return $this->order;
18     }
19 }

所有监听器都可以通过FilterOrderEvent的getOrder方法获得order对象。

调度事件

dispatcher的dispatch()方法通知监听给定的事件的所有监听器,有两个参数,一个是需要调度的事件名,另一个就是传给所有监听器的Event对象。

 1 use Acme\StoreBundle\StoreEvents;
 2 use Acme\StoreBundle\Order;
 3 use Acme\StoreBundle\Event\FilterOrderEvent;
 4
 5 // the order is somehow created or retrieved
 6 $order = new Order();
 7 // ...
 8
 9 // create the FilterOrderEvent and dispatch it
10 $event = new FilterOrderEvent($order);
11 $dispatcher->dispatch(StoreEvents::STORE_ORDER, $event);

    FilterOrderEvent对象作为参数传入到dispatch方法,现在,任何监听store.order事件的监听器都会接收到FilterOrderEvent对象,并通过调用getOrder方法获得order对象。

1 // some listener class that‘s been registered for "store.order" event
2 use Acme\StoreBundle\Event\FilterOrderEvent;
3
4 public function onStoreOrder(FilterOrderEvent $event)
5 {
6     $order = $event->getOrder();
7     // do something to or with the order
8 }

Event Subscribers

最普遍的监听事件的方法是注册一个监听器到dispatcher中,一个监听器可以监听一个或者多个事件。

还有另一种监听事件的方法是使用Event SubScriber,Event SubScriber是一个PHP类,能够准确的告诉dispatcher它订阅了那些事件。实现EventSubscriberInterface接口,该接口有一个静态的方法getSubscriberdEvents。

namespace Acme\StoreBundle\Event;

use Symfony\Component\EventDispatcher\EventSubscriberInterface;
use Symfony\Component\HttpKernel\Event\FilterResponseEvent;

class StoreSubscriber implements EventSubscriberInterface
{
    public static function getSubscribedEvents()
    {
        return array(
            ‘kernel.response‘ => array(
                array(‘onKernelResponsePre‘, 10),
                array(‘onKernelResponseMid‘, 5),
                array(‘onKernelResponsePost‘, 0),
            ),
            ‘store.order‘     => array(‘onStoreOrder‘, 0),
        );
    }

    public function onKernelResponsePre(FilterResponseEvent $event)
    {
        // ...
    }

    public function onKernelResponseMid(FilterResponseEvent $event)
    {
        // ...
    }

    public function onKernelResponsePost(FilterResponseEvent $event)
    {
        // ...
    }

    public function onStoreOrder(FilterOrderEvent $event)
    {
        // ...
    }
}

这个监听器类很简单,告诉了dispatcher监听了什么事件,还有监听的事件触发的方法。addSubscriber()方法把subscriber注册到dispatcher。

1 use Acme\StoreBundle\Event\StoreSubscriber;
2
3 $subscriber = new StoreSubscriber();
4 $dispatcher->addSubscriber($subscriber);

dispatcher准确的把Subscriber注册到EventSubscriberInterface::getSubscriberdEvents()返回的事件里,EventSubscriberInterface::getSubscriberdEvents()方法返回一个数组,数组的键对应Subscriber监听的事件,值对应这Subscriber处理该事件调用的一个方法或者一组方法。上面的例子中,一组监听器的方法对应这一个事件,同时我们也可以设置优先级来控制这组方法的执行先后顺序。当kernel.response事件被触发onKernelResponsePreonKernelResponseMid, 和 onKernelResponsePost三个方法就会先后执行。

停止事件的传递

在一些情况下,监听器可以停止事件传递下去,防止后续的监听器被调用,换句话说,监听器必须通知dispatcher停止传递事件给后续的监听器。在监听器里面实现stopPropagation()方法:

1 use Acme\StoreBundle\Event\FilterOrderEvent;
2
3 public function onStoreOrder(FilterOrderEvent $event)
4 {
5     // ...
6
7     $event->stopPropagation();
8 }

那么,监听了store.order事件的还没有执行的监听器就不会在被执行。

通过isPropagationStopped()方法可以判断一个事件是否被停止。

1 $dispatcher->dispatch(‘foo.event‘, $event);
2 if ($event->isPropagationStopped()) {
3     // ...
4 }
时间: 2024-10-19 14:18:55

Symfony2 EventDispatcher组件的相关文章

Symfony2学习笔记之事件分配器

----EventDispatcher组件使用 简介:       面向对象编程已经在确保代码的可扩展性方面走过了很长一段路.它是通过创建一些责任明确的类,让它们之间变得更加灵活,开发者可以通过继承这些类创建子类,来改变它们的行为.但是如果想将某个开发者的改变跟其它已经编写了自己子类的开发者共享,这种面向对象的继承就不再那么有用了. 举一个现实的实例,你想为你的项目提供一个插件系统.一个能够被添加到方法的插件,或者在方法执行的前后完成某些工作,而不会干扰到其它插件.这个通过单一的继承完成不是一个

Symfony2中的设计模式——装饰者模式

装饰者模式的定义 在不必改变原类文件和使用继承的情况下,动态地扩展一个对象的功能.它是通过创建一个包装对象,也就是装饰来包裹真实的对象. 装饰者模式把每个要装饰的功能放在单独的类中,并让这个类包装它要装饰的对象,因此,当需要执行特殊行为时,客户端代码就可以在运行的时候根据需要有选择地.按顺序地使用装饰功能包装对象了. 图1 使用场景 设想一下,如果我们需要创建一个在不同场合有不同着装的学生,例如:在学校学生需要穿上校服,在舞会学生需要穿上正装,在家学生可以裸装(有点变态),当然,还可以学习超人把

【转】从框架看PHP的五种境界及各自的薪资待遇

无意中看到这篇文章,有些触动,作为博客开篇,用来激励自己. 原文地址:点击打开 在撰写此文前首先必须申明的是本人不鄙视任何一种框架,也无意于挑起PHP框架间的战争,更没有贬低某个框架使用者的用意,本文纯粹个人的看法.你可以认为我无知也好,或者装逼也好,请不要试着在任何情况下,随便发起言语的攻击,这只是我个人的一些经验而已,你可以选择赞成或不赞成,也可以选择看与不看. 还有本人基本对这些框架中的大部分都用的时间比较长,绝对不是随便出个Hello World!就在这里评论,曾任某某底层本土公司的技术

国外程序员收集整理的PHP资源大全[转]

ziadoz在 Github发起维护的一个PHP资源列表,内容包括:库.框架.模板.安全.代码分析.日志.第三方库.配置工具.Web 工具.书籍.电子书.经典博文等等.伯乐在线对该资源列表进行了翻译,译文如下: 依赖管理 依赖和包管理库 Composer/ Packagist:一个包和依赖管理器 Composer Installers:一个多框架Composer库安装器 Pickle:一个PHP扩展安装器 其他的依赖管理 其他的相关依赖管理 Satis:一个静态Composer存储库生成器 Co

【转】国外程序员收集整理的PHP资源大全

其他框架 其他Web开发框架 Symfony CMF- 创建自定义CMS的内容管理框架 Knp RAD Bundle-Symfony 2的快速应用程序(RAD)包 框架组件 来自Web开发框架的独立组件 Symfony2 Components-Symfony2组件 Zend Framework 2 Components-ZF2组件 Aura Components-PHP5.4组件包 Hoa Project-另一个PHP组件包 微型框架 微型框架和路由 Silex- 基于Symfony2组件的微型

国外程序员收集整理的PHP资源大全

依赖管理 依赖和包管理库 Composer/ Packagist:一个包和依赖管理器 Composer Installers:一个多框架Composer库安装器 Pickle:一个PHP扩展安装器 其他的依赖管理 其他的相关依赖管理 Satis:一个静态Composer存储库生成器 Composition:一个在运行时检查Composer环境的库 Version:语义版本的解析和比较库 NameSpacer-转化下划线到命名空间的库 Patch Installer-使用Composer安装补丁的

从零开始学编程系列汇总

github 地址:https://github.com/justjavac/Programming-Alpha-To-Omega 「从零开始」系列 从零开始写个编译器吧系列 (知乎专栏) 从零开始写一个简单的操作系统 (伯乐在线) 从零开始写 JavaScript 框架 (图灵社区) 从零开始写 jQuery 框架 (蓝色理想 ) 从零开始 nodejs 系列文章 (粉丝日志) 从零开始编写网络游戏 ( 英特尔? 开发人员专区) 从零开始编写自己的 C# 框架 (博客园) 如何从零开始写一个

构建自己的PHP框架--搭建基本结构

首先,我们来说一下,为什么要创建自己的框架? 为什么要创建自己的框架呢?如果你跟周围的人讨论,每个人都会告诉你重复发明轮子是一件糟糕的事情,你最好选择一个已有的框架,忘掉“创建自己的框架”这种想法.大部分情况,他们是正确的,但是我想到了几个创建自己的框架的好处: 了解更多框架的底层架构 创建一个能满足你特殊需求的框架(但首先要确定你的需求真的是很特别) 因为乐趣而试着写一个框架(为了“学习然后抛弃”的目的) 想利用新的开发技术以及最佳实践重构已经存在的项目 向世界证明自己也是可以写出框架的(..

听 Fabien Potencier 谈Symfony2 之 《What is Symfony2 ?》

Symfoy2 是什么? PHP世界里又一广受关注的web MVC框架? Fabien Potencier 却不这么说! Fabien Potencier这样定义Symfoy2 是个什么东西: 首先,Symfony2 是一个独立,松散的,有组织严密的PHP组件集合,它可以为你解决一些web开发中遇到的一般性问题. 其次,基于这些组件,Symfoy2 也可以作为一个独立的web框架使用. 那么Symfony2 是一个MVC框架吗? Fabien Potencier 说Symfony2从来没有把自己