Symfony - 事件和 EventListener

Symfony 通过其 EventDispatcher 组件提供基于事件的编程。任何企业应用程序都需要基于事件的编程来创建高度可定制的应用程序。事件是对象之间交互的主要工具之一。没有事件,对象就无法有效地交互。

基于事件的编程过程可以概括为 - 一个名为 Event source 的对象要求中央调度程序对象注册一个事件,例如 user.registered。一个或多个对象(称为 listener)要求中央调度程序对象监听特定事件,例如 user.registered。在某个时间点,事件源对象要求中央调度程序对象调度事件,例如 user.registered 以及带有必要信息的事件对象。中央调度程序向所有侦听器对象通知有关事件的信息,例如 user.registered 及其 Event* 对象。

在基于事件的编程中,我们有四种类型的对象:事件源、事件侦听器、事件调度程序和事件本身。

让我们编写一个简单的应用程序来理解这个概念。

步骤 1 − 创建一个项目,event-dispatcher-example

cd /path/to/dir
mkdir event-dispatcher-example
cd event-dispatcher-example
composer require symfony/event-dispatcher

步骤 2 − 创建一个类,.User

class User { 
   public $name; 
   public $age; 
}  

$user = new User(); 
$user->name = "Jon"; 
$user->age = 25

步骤 3 − 创建一个事件 UserRegisteredEvent

use Symfony\Component\EventDispatcher\Event;  
class UserRegisteredEvent extends Event {
   const NAME = 'user.registered';  
   protected $user;  
   
   public function __construct(User $user) { 
      $this-<user = $user; 
   }  
   public function getUser() { 
      return $this-<user; 
   } 
}  
$event = new UserRegisteredEvent($user);

此处,UserRegisteredEvent 可以访问 User 对象。事件的名称为 user.registered

步骤 4 − 创建一个监听器,UserListener

class UserListener { 
   public function onUserRegistrationAction(Event $event) { 
      $user = $event->getUser(); 
         echo $user->name . "
"; 
         echo $user->age . "
"; 
   } 
}  
$listener = new UserListener(); 

步骤 5 − 创建事件调度器对象。

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

步骤 6 − 使用调度器对象及其方法 addListener 连接侦听器和事件。

$dispatcher 
   ->addListener( 
      UserRegisteredEvent::NAME,  
      array($listener, 'onUserRegistrationAction'));  

我们还可以添加一个匿名函数作为事件监听器,如下面的代码所示。

$dispatcher 
   ->addListener( 
      UserRegisteredEvent::NAME,  
      
      function(Event $event) { 
         $user = $event->getUser(); 
         echo $user->name . "
"; 
      }); 

步骤 7 − 最后,使用事件调度程序的方法 dispatch 触发/调度事件。

$dispatcher->dispatch(UserRegisteredEvent::NAME, $event);

完整代码清单如下。

main.php

<?php  
   require __DIR__ . '/vendor/autoload.php';  
   use Symfony\Component\EventDispatcher\EventDispatcher; 
   use Symfony\Component\EventDispatcher\Event;  

   class User { 
      public $name; 
      public $age; 
   }  
   class UserRegisteredEvent extends Event { 
      const NAME = 'user.registered';  
      protected $user;  
      
      public function __construct(User $user) { 
         $this->user = $user; 
      }  
      public function getUser() { 
         return $this->user; 
      } 
   }  
   class UserListener { 
      public function onUserRegistrationAction(Event $event) { 
         $user = $event->getUser(); 
         echo $user->name . "
"; 
         echo $user->age . "
"; 
      } 
   } 
   $user = new User(); 
   $user->name = "Jon"; 
   $user->age = 25;  

   $event = new UserRegisteredEvent($user); 
   $listener = new UserListener();  

   $dispatcher = new EventDispatcher();  
   $dispatcher 
      ->addListener( 
         UserRegisteredEvent::NAME, 
         
         function(Event $event) { 
            $user = $event->getUser(); 
            echo $user->name . "
"; 
         });  
   $dispatcher 
      ->addListener( 
         UserRegisteredEvent::NAME, array($listener, 'onUserRegistrationAction'));

   $dispatcher->dispatch(UserRegisteredEvent::NAME, $event); 
?>

结果

Jon 
Jon 
25

Symfony Web 框架有很多事件,可以为这些事件注册监听器并进行相应的编程。其中一个示例事件是 kernel.exception,对应的事件是 GetResponseForExceptionEvent,它保存响应对象(Web 请求的输出)。这用于捕获异常并使用通用错误信息修改响应,而不是向用户显示运行时错误。