深入解析PHP的Yii框架中的event事件机制
事件
事件可以将自定义代码“注入”到现有代码中的特定执行点。附加自定义代码到某个事件,当这个事件被触发时,这些代码就会自动执行。例如,邮件程序对象成功发出消息时可触发messageSent事件。如想追踪成功发送的消息,可以附加相应追踪代码到messageSent事件。
Yii引入了名为yii\base\Component的基类以支持事件。如果一个类需要触发事件就应该继承yii\base\Component或其子类。
Yii的event机制
YII的事件机制,是其比较独特之处,合理使用好事件机制,会使各个组件之间的耦合更为松散,利于团体协作开发。
何时需要使用事件,如何给事件绑定事件处理函数,以及如何触发事件,与其它语言是有较大的差别的。例如Javascript中,可以使用
$(‘#id').on("click",function(){});
方式给DOM元素绑定处理函数,当DOM元素上发生指定的事件(如click)时,将自动执行设定的函数。
但是PHP是服务器端的脚本语言,就不存在自动触发事件之说,所以和Javascript对比,YII中的事件是需要手动触发的。一般来说,要实现YII组件的事件机制,需要以下几步:
定义事件名称,其实就是级组件定义一个on开头的方法,其中的代码是固定的,如:
publicfunctiononBeginRequest($event){ $this->raiseEvent('onBeginRequest',$event); }
即函数名与事件名是一致的。此步的作用就是将绑定在此事件上的处理函数逐个执行。写这一系列的播客,算是一个整理,所以我写细一点,现在把raiseEvent方法的代码贴出来。
/***Raisesanevent. *Thismethodrepresentsthehappeningofanevent.Itinvokes *allattachedhandlersfortheevent. *@paramstring$nametheeventname *@paramCEvent$eventtheeventparameter *@throwsCExceptioniftheeventisundefinedoraneventhandlerisinvalid. */ publicfunctionraiseEvent($name,$event){ $name=strtolower($name); //_e这个数组用来存所有事件信息 if(isset($this->_e[$name])){ foreach($this->_e[$name]as$handler){ if(is_string($handler)) call_user_func($handler,$event); elseif(is_callable($handler,true)){ if(is_array($handler)){ //anarray:0-object,1-methodname list($object,$method)=$handler; if(is_string($object))//staticmethodcall call_user_func($handler,$event); elseif(method_exists($object,$method)) $object->$method($event); else thrownewCException(Yii::t('yii','Event"{class}.{event}"isattachedwithaninvalidhandler"{handler}".',array('{class}'=>get_class($this),'{event}'=>$name,'{handler}'=>$handler[1]))); } else//PHP5.3:anonymousfunction call_user_func($handler,$event); } else thrownewCException(Yii::t('yii','Event"{class}.{event}"isattachedwithaninvalidhandler"{handler}".',array('{class}'=>get_class($this),'{event}'=>$name,'{handler}'=>gettype($handler)))); //stopfurtherhandlingifparam.handledissettrue if(($eventinstanceofCEvent)&&$event->handled) return; } }elseif(YII_DEBUG&&!$this->hasEvent($name)) thrownewCException(Yii::t('yii','Event"{class}.{event}"isnotdefined.',array('{class}'=>get_class($this),'{event}'=>$name))); }
事件处理器(EventHandlers)
事件处理器是一个PHP回调函数,当它所附加到的事件被触发时它就会执行。可以使用以下回调函数之一:
- 字符串形式指定的PHP全局函数,如'trim';
- 对象名和方法名数组形式指定的对象方法,如[$object,$method];
- 类名和方法名数组形式指定的静态类方法,如[$class,$method];
- 匿名函数,如function($event){...}。
事件处理器的格式是:
function($event){ //$event是yii\base\Event或其子类的对象 }
通过$event参数,事件处理器就获得了以下有关事件的信息:
- yii\base\Event::name:事件名
- yii\base\Event::sender:调用trigger()方法的对象
- yii\base\Event::data:附加事件处理器时传入的数据,默认为空,后文详述
附加事件处理器
调用yii\base\Component::on()方法来附加处理器到事件上。如:
$foo=newFoo; //处理器是全局函数 $foo->on(Foo::EVENT_HELLO,'function_name'); //处理器是对象方法 $foo->on(Foo::EVENT_HELLO,[$object,'methodName']); //处理器是静态类方法 $foo->on(Foo::EVENT_HELLO,['app\components\Bar','methodName']); //处理器是匿名函数 $foo->on(Foo::EVENT_HELLO,function($event){ //事件处理逻辑 }); 附加事件处理器时可以提供额外数据作为yii\base\Component::on()方法的第三个参数。数据在事件被触发和处理器被调用时能被处理器使用。如: //当事件被触发时以下代码显示"abc" //因为$event->data包括被传递到"on"方法的数据 $foo->on(Foo::EVENT_HELLO,function($event){ echo$event->data; },'abc');
事件处理器顺序
可以附加一个或多个处理器到一个事件。当事件被触发,已附加的处理器将按附加次序依次调用。如果某个处理器需要停止其后的处理器调用,可以设置$event参数的[yii\base\Event::handled]]属性为真,如下:
$foo->on(Foo::EVENT_HELLO,function($event){ $event->handled=true; });
默认新附加的事件处理器排在已存在处理器队列的最后。因此,这个处理器将在事件被触发时最后一个调用。在处理器队列最前面插入新处理器将使该处理器最先调用,可以传递第四个参数$append为假并调用yii\base\Component::on()方法实现:
$foo->on(Foo::EVENT_HELLO,function($event){ //这个处理器将被插入到处理器队列的第一位... },$data,false);
触发事件
事件通过调用yii\base\Component::trigger()方法触发,此方法须传递事件名,还可以传递一个事件对象,用来传递参数到事件处理器。如:
namespaceapp\components; useyii\base\Component; useyii\base\Event; classFooextendsComponent { constEVENT_HELLO='hello'; publicfunctionbar() { $this->trigger(self::EVENT_HELLO); } }
以上代码当调用bar(),它将触发名为hello的事件。
提示:推荐使用类常量来表示事件名。上例中,常量EVENT_HELLO用来表示hello。这有两个好处。第一,它可以防止拼写错误并支持IDE的自动完成。第二,只要简单检查常量声明就能了解一个类支持哪些事件。
有时想要在触发事件时同时传递一些额外信息到事件处理器。例如,邮件程序要传递消息信息到messageSent事件的处理器以便处理器了解哪些消息被发送了。为此,可以提供一个事件对象作为yii\base\Component::trigger()方法的第二个参数。这个事件对象必须是yii\base\Event类或其子类的实例。如:
namespaceapp\components; useyii\base\Component; useyii\base\Event; classMessageEventextendsEvent { public$message; } classMailerextendsComponent { constEVENT_MESSAGE_SENT='messageSent'; publicfunctionsend($message) { //...发送$message的逻辑... $event=newMessageEvent; $event->message=$message; $this->trigger(self::EVENT_MESSAGE_SENT,$event); } }
当yii\base\Component::trigger()方法被调用时,它将调用所有附加到命名事件(trigger方法第一个参数)的事件处理器。
移除事件处理器
从事件移除处理器,调用yii\base\Component::off()方法。如:
//处理器是全局函数 $foo->off(Foo::EVENT_HELLO,'function_name'); //处理器是对象方法 $foo->off(Foo::EVENT_HELLO,[$object,'methodName']); //处理器是静态类方法 $foo->off(Foo::EVENT_HELLO,['app\components\Bar','methodName']); //处理器是匿名函数 $foo->off(Foo::EVENT_HELLO,$anonymousFunction);
注意当匿名函数附加到事件后一般不要尝试移除匿名函数,除非你在某处存储了它。以上示例中,假设匿名函数存储为变量$anonymousFunction。
移除事件的全部处理器,简单调用yii\base\Component::off()即可,不需要第二个参数:
$foo->off(Foo::EVENT_HELLO);
类级别的事件处理器
以上部分,我们叙述了在实例级别如何附加处理器到事件。有时想要一个类的所有实例而不是一个指定的实例都响应一个被触发的事件,并不是一个个附加事件处理器到每个实例,而是通过调用静态方法yii\base\Event::on()在类级别附加处理器。
例如,活动记录对象要在每次往数据库新增一条新记录时触发一个yii\db\BaseActiveRecord::EVENT_AFTER_INSERT事件。要追踪每个活动记录对象的新增记录完成情况,应如下写代码:
useYii; useyii\base\Event; useyii\db\ActiveRecord; Event::on(ActiveRecord::className(),ActiveRecord::EVENT_AFTER_INSERT,function($event){ Yii::trace(get_class($event->sender).'isinserted'); });
每当yii\db\BaseActiveRecord或其子类的实例触发yii\db\BaseActiveRecord::EVENT_AFTER_INSERT事件时,这个事件处理器都会执行。在这个处理器中,可以通过$event->sender获取触发事件的对象。
当对象触发事件时,它首先调用实例级别的处理器,然后才会调用类级别处理器。
可调用静态方法yii\base\Event::trigger()来触发一个类级别事件。类级别事件不与特定对象相关联。因此,它只会引起类级别事件处理器的调用。如:
useyii\base\Event; Event::on(Foo::className(),Foo::EVENT_HELLO,function($event){ echo$event->sender;//显示"app\models\Foo" }); Event::trigger(Foo::className(),Foo::EVENT_HELLO);
注意这种情况下$event->sender指向触发事件的类名而不是对象实例。
注意:因为类级别的处理器响应类和其子类的所有实例触发的事件,必须谨慎使用,尤其是底层的基类,如yii\base\Object。
移除类级别的事件处理器只需调用yii\base\Event::off(),如:
//移除$handler Event::off(Foo::className(),Foo::EVENT_HELLO,$handler); //移除Foo::EVENT_HELLO事件的全部处理器 Event::off(Foo::className(),Foo::EVENT_HELLO);
全局事件
所谓全局事件实际上是一个基于以上叙述的事件机制的戏法。它需要一个全局可访问的单例,如应用实例。
事件触发者不调用其自身的trigger()方法,而是调用单例的trigger()方法来触发全局事件。类似地,事件处理器被附加到单例的事件。如:
useYii; useyii\base\Event; useapp\components\Foo; Yii::$app->on('bar',function($event){ echoget_class($event->sender);//显示"app\components\Foo" }); Yii::$app->trigger('bar',newEvent(['sender'=>newFoo]));
全局事件的一个好处是当附加处理器到一个对象要触发的事件时,不需要产生该对象。相反,处理器附加和事件触发都通过单例(如应用实例)完成。
然而,因为全局事件的命名空间由各方共享,应合理命名全局事件,如引入一些命名空间(例:"frontend.mail.sent","backend.mail.sent")。
给组件对象绑定事件处理函数
$component->attachEventHandler($name,$handler); $component->onBeginRequest=$handler;
yii支持一个事件绑定多个回调函数,上述的两个方法都会在已有的事件上增加新的回调函数,而不会覆盖已有回调函数。
$handler即是一个PHP回调函数,关于回调函数的形式,本文的最后会附带说明。如CLogRouter组件的init事件中,有以下代码:
Yii::app()->attachEventHandler('onEndRequest',array($this,'processLogs'));
这就是给CApplication对象的onEndRequest绑定了CLogRouter::processLogs()回调函数。而CApplication组件确实存在名为onEndRequest的方法(即onEndRequest事件),它之中的代码就是激活了相应的回调函数,即CLogRouter::processLogs()方法。所以从这里可以得出,日志的记录其实是发生在CApplication组件的正常退出时。
在需要触发事件的时候,直接激活组件的事件,即调用事件即可,如:比如CApplication组件的run方法中:
if($this->hasEventHandler('onBeginRequest')) $this->onBeginRequest(newCEvent($this));
这样即触发了事件处理函数。如果没有第一行的判断,那么在调试模式下(YII_DEBUG常量被定义为true),会抛出异常,而在非调试模式下(YII_DEBUG常量定义为false或没有定义YII_DEBUG常量),则不会产生任何异常。
回调函数的形式:
普通全局函数(内置的或用户自定义的)
call_user_func(‘print',$str);
类的静态方法,使用数组形式传递
call_user_func(array(‘className',‘print'),$str);
对象方法,使用数组形式传递
$obj=newclassName(); call_user_func(array($obj,‘print'),$str);
匿名方法,类似javascript的匿名函数
call_user_func(function($i){echo$i++;},4);
或使用以下形式:
$s=function($i){ echo$i++; }; call_user_func($s,4);
总结
关于Yii的事件机制其实就是提供了一种用于解耦的方式,在需要调用event的地方之前,只要你提供了事件的实现并注册在之后的地方需要的时候即可调用。