发布于 2015-08-27 16:52:33 | 233 次阅读 | 评论: 0 | 来源: 网络整理
The EventDispatcher component provides tools that allow your application components to communicate with each other by dispatching events and listening to them.
Object Oriented code has gone a long way to ensuring code extensibility. By creating classes that have well defined responsibilities, your code becomes more flexible and a developer can extend them with subclasses to modify their behaviors. But if they want to share the changes with other developers who have also made their own subclasses, code inheritance is no longer the answer.
Consider the real-world example where you want to provide a plugin system for your project. A plugin should be able to add methods, or do something before or after a method is executed, without interfering with other plugins. This is not an easy problem to solve with single inheritance, and multiple inheritance (were it possible with PHP) has its own drawbacks.
The Symfony EventDispatcher component implements the Mediator pattern in a simple and effective way to make all these things possible and to make your projects truly extensible.
Take a simple example from The HttpKernel Component. Once a
Response
object has been created, it may be useful to allow other elements
in the system to modify it (e.g. add some cache headers) before it’s actually
used. To make this possible, the Symfony kernel throws an event -
kernel.response
. Here’s how it works:
kernel.response
event;kernel.response
event, passing with it an Event
object that has
access to the Response
object;kernel.response
event, allowing each of them to make modifications to
the Response
object.You can install the component in 2 different ways:
symfony/event-dispatcher
on Packagist);When an event is dispatched, it’s identified by a unique name (e.g.
kernel.response
), which any number of listeners might be listening to. An
Event
instance is also created
and passed to all of the listeners. As you’ll see later, the Event
object
itself often contains data about the event being dispatched.
The unique event name can be any string, but optionally follows a few simple naming conventions:
.
), and underscores (_
);kernel.
);request
).Here are some examples of good event names:
kernel.response
form.pre_set_data
When the dispatcher notifies listeners, it passes an actual Event
object
to those listeners. The base Event
class is very simple: it contains a
method for stopping event
propagation, but not much else.
Often times, data about a specific event needs to be passed along with the
Event
object so that the listeners have needed information. In the case of
the kernel.response
event, the Event
object that’s created and passed to
each listener is actually of type
FilterResponseEvent
, a
subclass of the base Event
object. This class contains methods such as
getResponse
and setResponse
, allowing listeners to get or even replace
the Response
object.
The moral of the story is this: When creating a listener to an event, the
Event
object that’s passed to the listener may be a special subclass that
has additional methods for retrieving information from and responding to the
event.
The dispatcher is the central object of the event dispatcher system. In general, a single dispatcher is created, which maintains a registry of listeners. When an event is dispatched via the dispatcher, it notifies all listeners registered with that event:
use SymfonyComponentEventDispatcherEventDispatcher;
$dispatcher = new EventDispatcher();
To take advantage of an existing event, you need to connect a listener to the
dispatcher so that it can be notified when the event is dispatched. A call to
the dispatcher’s addListener()
method associates any valid PHP callable to
an event:
$listener = new AcmeListener();
$dispatcher->addListener('foo.action', array($listener, 'onFooAction'));
The addListener()
method takes up to three arguments:
0
). If two listeners
have the same priority, they are executed in the order that they were added
to the dispatcher.注解
A PHP callable is a PHP variable that can be used by the
call_user_func()
function and returns true
when passed to the
is_callable()
function. It can be a Closure
instance, an object
implementing an __invoke
method (which is what closures are in fact),
a string representing a function, or an array representing an object
method or a class method.
So far, you’ve seen how PHP objects can be registered as listeners. You can also register PHP Closures as event listeners:
use SymfonyComponentEventDispatcherEvent;
$dispatcher->addListener('foo.action', function (Event $event) {
// will be executed when the foo.action event is dispatched
});
Once a listener is registered with the dispatcher, it waits until the event is
notified. In the above example, when the foo.action
event is dispatched,
the dispatcher calls the AcmeListener::onFooAction
method and passes the
Event
object as the single argument:
use SymfonyComponentEventDispatcherEvent;
class AcmeListener
{
// ...
public function onFooAction(Event $event)
{
// ... do something
}
}
In many cases, a special Event
subclass that’s specific to the given event
is passed to the listener. This gives the listener access to special
information about the event. Check the documentation or implementation of each
event to determine the exact SymfonyComponentEventDispatcherEvent
instance that’s being passed. For example, the kernel.response
event passes an
instance of SymfonyComponentHttpKernelEventFilterResponseEvent
:
use SymfonyComponentHttpKernelEventFilterResponseEvent;
public function onKernelResponse(FilterResponseEvent $event)
{
$response = $event->getResponse();
$request = $event->getRequest();
// ...
}
In addition to registering listeners with existing events, you can create and dispatch your own events. This is useful when creating third-party libraries and also when you want to keep different components of your own system flexible and decoupled.
Events
Class¶Suppose you want to create a new Event - store.order
- that is dispatched
each time an order is created inside your application. To keep things
organized, start by creating a StoreEvents
class inside your application
that serves to define and document your event:
namespace AcmeStoreBundle;
final class StoreEvents
{
/**
* The store.order event is thrown each time an order is created
* in the system.
*
* The event listener receives an
* AcmeStoreBundleEventFilterOrderEvent instance.
*
* @var string
*/
const STORE_ORDER = 'store.order';
}
Notice that this class doesn’t actually do anything. The purpose of the
StoreEvents
class is just to be a location where information about common
events can be centralized. Notice also that a special FilterOrderEvent
class will be passed to each listener of this event.
Later, when you dispatch this new event, you’ll create an Event
instance
and pass it to the dispatcher. The dispatcher then passes this same instance
to each of the listeners of the event. If you don’t need to pass any
information to your listeners, you can use the default
SymfonyComponentEventDispatcherEvent
class. Most of the time, however,
you will need to pass information about the event to each listener. To
accomplish this, you’ll create a new class that extends
SymfonyComponentEventDispatcherEvent
.
In this example, each listener will need access to some pretend Order
object. Create an Event
class that makes this possible:
namespace AcmeStoreBundleEvent;
use SymfonyComponentEventDispatcherEvent;
use AcmeStoreBundleOrder;
class FilterOrderEvent extends Event
{
protected $order;
public function __construct(Order $order)
{
$this->order = $order;
}
public function getOrder()
{
return $this->order;
}
}
Each listener now has access to the Order
object via the getOrder
method.
The dispatch()
method notifies all listeners of the given event. It takes two arguments: the
name of the event to dispatch and the Event
instance to pass to each
listener of that event:
use AcmeStoreBundleStoreEvents;
use AcmeStoreBundleOrder;
use AcmeStoreBundleEventFilterOrderEvent;
// the order is somehow created or retrieved
$order = new Order();
// ...
// create the FilterOrderEvent and dispatch it
$event = new FilterOrderEvent($order);
$dispatcher->dispatch(StoreEvents::STORE_ORDER, $event);
Notice that the special FilterOrderEvent
object is created and passed to
the dispatch
method. Now, any listener to the store.order
event will
receive the FilterOrderEvent
and have access to the Order
object via
the getOrder
method:
// some listener class that's been registered for "store.order" event
use AcmeStoreBundleEventFilterOrderEvent;
public function onStoreOrder(FilterOrderEvent $event)
{
$order = $event->getOrder();
// do something to or with the order
}
The most common way to listen to an event is to register an event listener with the dispatcher. This listener can listen to one or more events and is notified each time those events are dispatched.
Another way to listen to events is via an event subscriber. An event
subscriber is a PHP class that’s able to tell the dispatcher exactly which
events it should subscribe to. It implements the
EventSubscriberInterface
interface, which requires a single static method called
getSubscribedEvents
. Take the following example of a subscriber that
subscribes to the kernel.response
and store.order
events:
namespace AcmeStoreBundleEvent;
use SymfonyComponentEventDispatcherEventSubscriberInterface;
use SymfonyComponentHttpKernelEventFilterResponseEvent;
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)
{
// ...
}
}
This is very similar to a listener class, except that the class itself can
tell the dispatcher which events it should listen to. To register a subscriber
with the dispatcher, use the
addSubscriber()
method:
use AcmeStoreBundleEventStoreSubscriber;
$subscriber = new StoreSubscriber();
$dispatcher->addSubscriber($subscriber);
The dispatcher will automatically register the subscriber for each event
returned by the getSubscribedEvents
method. This method returns an array
indexed by event names and whose values are either the method name to call or
an array composed of the method name to call and a priority. The example
above shows how to register several listener methods for the same event in
subscriber and also shows how to pass the priority of each listener method.
The higher the priority, the earlier the method is called. In the above
example, when the kernel.response
event is triggered, the methods
onKernelResponsePre
, onKernelResponseMid
, and onKernelResponsePost
are called in that order.
In some cases, it may make sense for a listener to prevent any other listeners
from being called. In other words, the listener needs to be able to tell the
dispatcher to stop all propagation of the event to future listeners (i.e. to
not notify any more listeners). This can be accomplished from inside a
listener via the
stopPropagation()
method:
use AcmeStoreBundleEventFilterOrderEvent;
public function onStoreOrder(FilterOrderEvent $event)
{
// ...
$event->stopPropagation();
}
Now, any listeners to store.order
that have not yet been called will not
be called.
It is possible to detect if an event was stopped by using the
isPropagationStopped()
method
which returns a boolean value:
$dispatcher->dispatch('foo.event', $event);
if ($event->isPropagationStopped()) {
// ...
}
The EventDispatcher
always passes the dispatched event, the event’s name
and a reference to itself to the listeners. This can be used in some advanced
usages of the EventDispatcher
like dispatching other events in listeners,
event chaining or even lazy loading of more listeners into the dispatcher
object as shown in the following examples.
Lazy loading listeners:
use SymfonyComponentEventDispatcherEvent;
use SymfonyComponentEventDispatcherEventDispatcherInterface;
use AcmeStoreBundleEventStoreSubscriber;
class Foo
{
private $started = false;
public function myLazyListener(
Event $event,
$eventName,
EventDispatcherInterface $dispatcher
) {
if (false === $this->started) {
$subscriber = new StoreSubscriber();
$dispatcher->addSubscriber($subscriber);
}
$this->started = true;
// ... more code
}
}
Dispatching another event from within a listener:
use SymfonyComponentEventDispatcherEvent;
use SymfonyComponentEventDispatcherEventDispatcherInterface;
class Foo
{
public function myFooListener(
Event $event,
$eventName,
EventDispatcherInterface $dispatcher
) {
$dispatcher->dispatch('log', $event);
// ... more code
}
}
While this above is sufficient for most uses, if your application uses multiple
EventDispatcher
instances, you might need to specifically inject a known
instance of the EventDispatcher
into your listeners. This could be done
using constructor or setter injection as follows:
Constructor injection:
use SymfonyComponentEventDispatcherEventDispatcherInterface;
class Foo
{
protected $dispatcher = null;
public function __construct(EventDispatcherInterface $dispatcher)
{
$this->dispatcher = $dispatcher;
}
}
Or setter injection:
use SymfonyComponentEventDispatcherEventDispatcherInterface;
class Foo
{
protected $dispatcher = null;
public function setEventDispatcher(EventDispatcherInterface $dispatcher)
{
$this->dispatcher = $dispatcher;
}
}
Choosing between the two is really a matter of taste. Many tend to prefer the constructor injection as the objects are fully initialized at construction time. But when you have a long list of dependencies, using setter injection can be the way to go, especially for optional dependencies.
The EventDispatcher::dispatch
method always returns an Event
object. This allows for various shortcuts. For example, if one does not need
a custom event object, one can simply rely on a plain
Event
object. You do not even need
to pass this to the dispatcher as it will create one by default unless you
specifically pass one:
$dispatcher->dispatch('foo.event');
Moreover, the EventDispatcher always returns whichever event object that was dispatched, i.e. either the event that was passed or the event that was created internally by the dispatcher. This allows for nice shortcuts:
if (!$dispatcher->dispatch('foo.event')->isPropagationStopped()) {
// ...
}
Or:
$barEvent = new BarEvent();
$bar = $dispatcher->dispatch('bar.event', $barEvent)->getBar();
Or:
$bar = $dispatcher->dispatch('bar.event', new BarEvent())->getBar();
and so on...
2.4 新版功能: Before Symfony 2.4, the event name and the event dispatcher had to be
requested from the Event
instance. These methods are now deprecated.
The EventDispatcher
instance, as well as the name of the event that is
dispatched, are passed as arguments to the listener:
use SymfonyComponentEventDispatcherEvent;
use SymfonyComponentEventDispatcherEventDispatcherInterface;
class Foo
{
public function myEventListener(Event $event, $eventName, EventDispatcherInterface $dispatcher)
{
echo $eventName;
}
}
Besides the commonly used EventDispatcher
, the component comes with 2
other dispatchers: