国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院

首頁(yè) > 開(kāi)發(fā) > PHP > 正文

深入解析PHP的Yii框架中的event事件機(jī)制

2024-05-04 23:44:04
字體:
來(lái)源:轉(zhuǎn)載
供稿:網(wǎng)友
這篇文章主要介紹了PHP的Yii框架中的event事件機(jī)制,文中講解了Yii的事件處理器以及給組件對(duì)象綁定事件處理函數(shù)等重要知識(shí),需要的朋友可以參考下
 

事件
事件可以將自定義代碼“注入”到現(xiàn)有代碼中的特定執(zhí)行點(diǎn)。附加自定義代碼到某個(gè)事件,當(dāng)這個(gè)事件被觸發(fā)時(shí),這些代碼就會(huì)自動(dòng)執(zhí)行。例如,郵件程序?qū)ο蟪晒Πl(fā)出消息時(shí)可觸發(fā) messageSent 事件。如想追蹤成功發(fā)送的消息,可以附加相應(yīng)追蹤代碼到messageSent 事件。
Yii 引入了名為 yii/base/Component 的基類(lèi)以支持事件。如果一個(gè)類(lèi)需要觸發(fā)事件就應(yīng)該繼承 yii/base/Component 或其子類(lèi)。

Yii的event機(jī)制
YII的事件機(jī)制,是其比較獨(dú)特之處,合理使用好事件機(jī)制,會(huì)使各個(gè)組件之間的耦合更為松散,利于團(tuán)體協(xié)作開(kāi)發(fā)。
何時(shí)需要使用事件,如何給事件綁定事件處理函數(shù),以及如何觸發(fā)事件,與其它語(yǔ)言是有較大的差別的。例如Javascript中,可以使用

$(‘#id').on("click",function() {});

方式給DOM元素綁定處理函數(shù),當(dāng)DOM元素上發(fā)生指定的事件(如click)時(shí),將自動(dòng)執(zhí)行設(shè)定的函數(shù)。 
但是PHP是服務(wù)器端的腳本語(yǔ)言,就不存在自動(dòng)觸發(fā)事件之說(shuō),所以和Javascript對(duì)比,YII中的事件是需要手動(dòng)觸發(fā)的。一般來(lái)說(shuō),要實(shí)現(xiàn)YII組件的事件機(jī)制,需要以下幾步:

定義事件名稱(chēng),其實(shí)就是級(jí)組件定義一個(gè)on開(kāi)頭的方法,其中的代碼是固定的,如:

 public function onBeginRequest($event){ $this->raiseEvent('onBeginRequest',$event);}

即函數(shù)名與事件名是一致的。此步的作用就是將綁定在此事件上的處理函數(shù)逐個(gè)執(zhí)行。寫(xiě)這一系列的播客,算是一個(gè)整理,所以我寫(xiě)細(xì)一點(diǎn),現(xiàn)在把raiseEvent方法的代碼貼出來(lái)。

/** * Raises an event.   * This method represents the happening of an event. It invokes   * all attached handlers for the event.   * @param string $name the event name   * @param CEvent $event the event parameter   * @throws CException if the event is undefined or an event handler is invalid. */  public function raiseEvent($name,$event){         $name=strtolower($name);         //_e這個(gè)數(shù)組用來(lái)存所有事件信息       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)){                           // an array: 0 - object, 1 - method name                     list($object,$method)=$handler;                      if(is_string($object)) // static method call                     call_user_func($handler,$event);                    elseif(method_exists($object,$method))                          $object->$method($event);                           else                              throw new CException(Yii::t('yii','Event "{class}.{event}" is attached with an invalid handler "{handler}".',  array('{class}'=>get_class($this), '{event}'=>$name, '{handler}'=>$handler[1])));                        }                        else // PHP 5.3: anonymous function                      call_user_func($handler,$event);                 }                 else                    throw new CException(Yii::t('yii','Event "{class}.{event}" is attached with an invalid handler "{handler}".', array('{class}'=>get_class($this), '{event}'=>$name, '{handler}'=>gettype($handler))));                // stop further handling if param.handled is set true             if(($event instanceof CEvent) && $event->handled)               return;             }         }  elseif(YII_DEBUG && !$this->hasEvent($name))             throw new CException(Yii::t('yii','Event "{class}.{event}" is not defined.',     array('{class}'=>get_class($this), '{event}'=>$name))); }

事件處理器(Event Handlers)

事件處理器是一個(gè)PHP 回調(diào)函數(shù),當(dāng)它所附加到的事件被觸發(fā)時(shí)它就會(huì)執(zhí)行。可以使用以下回調(diào)函數(shù)之一:

  • 字符串形式指定的 PHP 全局函數(shù),如 'trim' ;
  • 對(duì)象名和方法名數(shù)組形式指定的對(duì)象方法,如 [$object, $method] ;
  • 類(lèi)名和方法名數(shù)組形式指定的靜態(tài)類(lèi)方法,如 [$class, $method] ;
  • 匿名函數(shù),如 function ($event) { ... } 。

事件處理器的格式是:

function ($event) {  // $event 是 yii/base/Event 或其子類(lèi)的對(duì)象}

通過(guò) $event 參數(shù),事件處理器就獲得了以下有關(guān)事件的信息:

  • yii/base/Event::name:事件名
  • yii/base/Event::sender:調(diào)用 trigger() 方法的對(duì)象
  • yii/base/Event::data:附加事件處理器時(shí)傳入的數(shù)據(jù),默認(rèn)為空,后文詳述

附加事件處理器

調(diào)用 yii/base/Component::on() 方法來(lái)附加處理器到事件上。如:

$foo = new Foo;// 處理器是全局函數(shù)$foo->on(Foo::EVENT_HELLO, 'function_name');// 處理器是對(duì)象方法$foo->on(Foo::EVENT_HELLO, [$object, 'methodName']);// 處理器是靜態(tài)類(lèi)方法$foo->on(Foo::EVENT_HELLO, ['app/components/Bar', 'methodName']);// 處理器是匿名函數(shù)$foo->on(Foo::EVENT_HELLO, function ($event) {  //事件處理邏輯});附加事件處理器時(shí)可以提供額外數(shù)據(jù)作為 yii/base/Component::on() 方法的第三個(gè)參數(shù)。數(shù)據(jù)在事件被觸發(fā)和處理器被調(diào)用時(shí)能被處理器使用。如:// 當(dāng)事件被觸發(fā)時(shí)以下代碼顯示 "abc"// 因?yàn)?$event->data 包括被傳遞到 "on" 方法的數(shù)據(jù)$foo->on(Foo::EVENT_HELLO, function ($event) {  echo $event->data;}, 'abc');

事件處理器順序

可以附加一個(gè)或多個(gè)處理器到一個(gè)事件。當(dāng)事件被觸發(fā),已附加的處理器將按附加次序依次調(diào)用。如果某個(gè)處理器需要停止其后的處理器調(diào)用,可以設(shè)置 $event 參數(shù)的 [yii/base/Event::handled]] 屬性為真,如下:

$foo->on(Foo::EVENT_HELLO, function ($event) {  $event->handled = true;});

默認(rèn)新附加的事件處理器排在已存在處理器隊(duì)列的最后。因此,這個(gè)處理器將在事件被觸發(fā)時(shí)最后一個(gè)調(diào)用。在處理器隊(duì)列最前面插入新處理器將使該處理器最先調(diào)用,可以傳遞第四個(gè)參數(shù) $append 為假并調(diào)用 yii/base/Component::on() 方法實(shí)現(xiàn):

$foo->on(Foo::EVENT_HELLO, function ($event) {  // 這個(gè)處理器將被插入到處理器隊(duì)列的第一位...}, $data, false);


觸發(fā)事件

事件通過(guò)調(diào)用 yii/base/Component::trigger() 方法觸發(fā),此方法須傳遞事件名,還可以傳遞一個(gè)事件對(duì)象,用來(lái)傳遞參數(shù)到事件處理器。如:

namespace app/components;use yii/base/Component;use yii/base/Event;class Foo extends Component{  const EVENT_HELLO = 'hello';  public function bar()  {    $this->trigger(self::EVENT_HELLO);  }}

以上代碼當(dāng)調(diào)用 bar() ,它將觸發(fā)名為 hello 的事件。

提示:推薦使用類(lèi)常量來(lái)表示事件名。上例中,常量 EVENT_HELLO 用來(lái)表示 hello 。這有兩個(gè)好處。第一,它可以防止拼寫(xiě)錯(cuò)誤并支持 IDE 的自動(dòng)完成。第二,只要簡(jiǎn)單檢查常量聲明就能了解一個(gè)類(lèi)支持哪些事件。
有時(shí)想要在觸發(fā)事件時(shí)同時(shí)傳遞一些額外信息到事件處理器。例如,郵件程序要傳遞消息信息到 messageSent 事件的處理器以便處理器了解哪些消息被發(fā)送了。為此,可以提供一個(gè)事件對(duì)象作為 yii/base/Component::trigger() 方法的第二個(gè)參數(shù)。這個(gè)事件對(duì)象必須是 yii/base/Event 類(lèi)或其子類(lèi)的實(shí)例。如:

namespace app/components;use yii/base/Component;use yii/base/Event;class MessageEvent extends Event{  public $message;}class Mailer extends Component{  const EVENT_MESSAGE_SENT = 'messageSent';  public function send($message)  {    // ...發(fā)送 $message 的邏輯...    $event = new MessageEvent;    $event->message = $message;    $this->trigger(self::EVENT_MESSAGE_SENT, $event);  }}

當(dāng) yii/base/Component::trigger() 方法被調(diào)用時(shí),它將調(diào)用所有附加到命名事件(trigger 方法第一個(gè)參數(shù))的事件處理器。

移除事件處理器

從事件移除處理器,調(diào)用 yii/base/Component::off() 方法。如:

// 處理器是全局函數(shù)$foo->off(Foo::EVENT_HELLO, 'function_name');// 處理器是對(duì)象方法$foo->off(Foo::EVENT_HELLO, [$object, 'methodName']);// 處理器是靜態(tài)類(lèi)方法$foo->off(Foo::EVENT_HELLO, ['app/components/Bar', 'methodName']);// 處理器是匿名函數(shù)$foo->off(Foo::EVENT_HELLO, $anonymousFunction);

注意當(dāng)匿名函數(shù)附加到事件后一般不要嘗試移除匿名函數(shù),除非你在某處存儲(chǔ)了它。以上示例中,假設(shè)匿名函數(shù)存儲(chǔ)為變量$anonymousFunction 。

移除事件的全部處理器,簡(jiǎn)單調(diào)用 yii/base/Component::off() 即可,不需要第二個(gè)參數(shù):

$foo->off(Foo::EVENT_HELLO);

類(lèi)級(jí)別的事件處理器

以上部分,我們敘述了在實(shí)例級(jí)別如何附加處理器到事件。有時(shí)想要一個(gè)類(lèi)的所有實(shí)例而不是一個(gè)指定的實(shí)例都響應(yīng)一個(gè)被觸發(fā)的事件,并不是一個(gè)個(gè)附加事件處理器到每個(gè)實(shí)例,而是通過(guò)調(diào)用靜態(tài)方法 yii/base/Event::on() 在類(lèi)級(jí)別附加處理器。

例如,活動(dòng)記錄對(duì)象要在每次往數(shù)據(jù)庫(kù)新增一條新記錄時(shí)觸發(fā)一個(gè) yii/db/BaseActiveRecord::EVENT_AFTER_INSERT 事件。要追蹤每個(gè)活動(dòng)記錄對(duì)象的新增記錄完成情況,應(yīng)如下寫(xiě)代碼:

use Yii;use yii/base/Event;use yii/db/ActiveRecord;Event::on(ActiveRecord::className(), ActiveRecord::EVENT_AFTER_INSERT, function ($event) {  Yii::trace(get_class($event->sender) . ' is inserted');});

每當(dāng) yii/db/BaseActiveRecord 或其子類(lèi)的實(shí)例觸發(fā) yii/db/BaseActiveRecord::EVENT_AFTER_INSERT 事件時(shí),這個(gè)事件處理器都會(huì)執(zhí)行。在這個(gè)處理器中,可以通過(guò) $event->sender 獲取觸發(fā)事件的對(duì)象。

當(dāng)對(duì)象觸發(fā)事件時(shí),它首先調(diào)用實(shí)例級(jí)別的處理器,然后才會(huì)調(diào)用類(lèi)級(jí)別處理器。

可調(diào)用靜態(tài)方法yii/base/Event::trigger()來(lái)觸發(fā)一個(gè)類(lèi)級(jí)別事件。類(lèi)級(jí)別事件不與特定對(duì)象相關(guān)聯(lián)。因此,它只會(huì)引起類(lèi)級(jí)別事件處理器的調(diào)用。如:

use yii/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 指向觸發(fā)事件的類(lèi)名而不是對(duì)象實(shí)例。

注意:因?yàn)轭?lèi)級(jí)別的處理器響應(yīng)類(lèi)和其子類(lèi)的所有實(shí)例觸發(fā)的事件,必須謹(jǐn)慎使用,尤其是底層的基類(lèi),如 yii/base/Object。
移除類(lèi)級(jí)別的事件處理器只需調(diào)用yii/base/Event::off(),如:

// 移除 $handlerEvent::off(Foo::className(), Foo::EVENT_HELLO, $handler);// 移除 Foo::EVENT_HELLO 事件的全部處理器Event::off(Foo::className(), Foo::EVENT_HELLO);

全局事件

所謂全局事件實(shí)際上是一個(gè)基于以上敘述的事件機(jī)制的戲法。它需要一個(gè)全局可訪問(wèn)的單例,如應(yīng)用實(shí)例。

事件觸發(fā)者不調(diào)用其自身的 trigger() 方法,而是調(diào)用單例的 trigger() 方法來(lái)觸發(fā)全局事件。類(lèi)似地,事件處理器被附加到單例的事件。如:

use Yii;use yii/base/Event;use app/components/Foo;Yii::$app->on('bar', function ($event) {  echo get_class($event->sender); // 顯示 "app/components/Foo"});Yii::$app->trigger('bar', new Event(['sender' => new Foo]));

全局事件的一個(gè)好處是當(dāng)附加處理器到一個(gè)對(duì)象要觸發(fā)的事件時(shí),不需要產(chǎn)生該對(duì)象。相反,處理器附加和事件觸發(fā)都通過(guò)單例(如應(yīng)用實(shí)例)完成。

然而,因?yàn)槿质录拿臻g由各方共享,應(yīng)合理命名全局事件,如引入一些命名空間(例:"frontend.mail.sent", "backend.mail.sent")。

給組件對(duì)象綁定事件處理函數(shù)

$component->attachEventHandler($name, $handler);$component->onBeginRequest = $handler ;

yii支持一個(gè)事件綁定多個(gè)回調(diào)函數(shù),上述的兩個(gè)方法都會(huì)在已有的事件上增加新的回調(diào)函數(shù),而不會(huì)覆蓋已有回調(diào)函數(shù)。
$handler即是一個(gè)PHP回調(diào)函數(shù),關(guān)于回調(diào)函數(shù)的形式,本文的最后會(huì)附帶說(shuō)明。如CLogRouter組件的init事件中,有以下代碼:

Yii::app()->attachEventHandler('onEndRequest',array($this,'processLogs'));

這就是給CApplication對(duì)象的onEndRequest綁定了CLogRouter::processLogs()回調(diào)函數(shù)。而CApplication組件確實(shí)存在名為onEndRequest的方法(即onEndRequest事件),它之中的代碼就是激活了相應(yīng)的回調(diào)函數(shù),即CLogRouter::processLogs()方法。所以從這里可以得出,日志的記錄其實(shí)是發(fā)生在CApplication組件的正常退出時(shí)。

在需要觸發(fā)事件的時(shí)候,直接激活組件的事件,即調(diào)用事件即可,如:比如CApplication組件的run方法中:

if($this->hasEventHandler('onBeginRequest'))  $this->onBeginRequest(new CEvent($this));

這樣即觸發(fā)了事件處理函數(shù)。如果沒(méi)有第一行的判斷,那么在調(diào)試模式下(YII_DEBUG常量被定義為true),會(huì)拋出異常,而在非調(diào)試模式下(YII_DEBUG常量定義為false或沒(méi)有定義YII_DEBUG常量),則不會(huì)產(chǎn)生任何異常。
回調(diào)函數(shù)的形式:

普通全局函數(shù)(內(nèi)置的或用戶(hù)自定義的)

call_user_func(‘print', $str);

類(lèi)的靜態(tài)方法,使用數(shù)組形式傳遞

call_user_func(array(‘className', ‘print'), $str );

對(duì)象方法,使用數(shù)組形式傳遞

$obj = new className();call_user_func(array($obj, ‘print'), $str );

匿名方法,類(lèi)似javascript的匿名函數(shù)

call_user_func(function($i){echo $i++;},4);

或使用以下形式:

$s = function($i) {  echo $i++;};call_user_func($s,4);

總結(jié)

關(guān)于Yii的事件機(jī)制其實(shí)就是提供了一種用于解耦的方式,在需要調(diào)用event的地方之前,只要你提供了事件的實(shí)現(xiàn)并注冊(cè)在之后的地方需要的時(shí)候即可調(diào)用。



發(fā)表評(píng)論 共有條評(píng)論
用戶(hù)名: 密碼:
驗(yàn)證碼: 匿名發(fā)表
国产激情自拍_国产9色视频_丁香花在线电影小说观看 _久久久久国产精品嫩草影院
国产三级免费观看| 伊人网在线视频| 黄色片大全在线观看| 国产蜜臀av在线播放| 欧美色第一页| 成年人在线观看| 国产精品一区二区三区高清在线 | 国产黄网站在线观看| 国产福利在线播放麻豆| 欧美专区日韩| 国产成人综合亚洲欧美在| 丁香视频五月| 成在在线免费视频| 国产麻豆精品视频一区二区| 亚洲第一区视频| h网址在线观看| 国产美女一区视频| 最近免费中文字幕大全免费第三页| 国产一级二级在线| 999福利在线视频| 日本高清不卡中文字幕| 中中文字幕av在线| av免费在线一区二区三区| 成在线人视频免费视频| 国产h色视频在线观看| 天堂中文在线观看| 最近中文字幕av免费高清| 精品女厕厕露p撒尿| 精品国产福利一区二区在线| 在线免费观看黄色片| 97在线超碰| 99热在线观看免费| av亚洲男人天堂| 国产福利在线| 国产性色视频| 国产女人在线视频| 免费在线看v| 在线观看中文字幕的网站| 国产视频中文字幕在线观看| 日本高清不卡中文字幕| 国产a级网站| 国产三级在线播放| 国产天堂在线| 国产免费av网站| 日本成人免费网站| www操操操| 日本福利在线观看| 欧美另类在线视频| 免费不卡中文字幕视频| 色中文字幕在线| 亚洲免费国产| 日本久久国产| 91涩漫在线观看c| 九九在线视频| 超碰免费在线观看| 国产综合视频一区二区三区免费| www.eeuss影院| 久久久久久91精品色婷婷| 亚洲免费网站在线观看| 国产成人午夜精品| 国产成人精品久久一区二区小说| 97在线超碰| 青青草免费在线视频| 四虎国产精品永久地址998| a视频在线播放| 超碰免费在线播放| 中文字幕在线免费| 精品福利影院| 超碰免费在线播放| 国产人成精品| 免费a级在线播放| 国产精品ⅴa有声小说| 亚洲天堂影院在线观看| 久久五月精品| 国产视频中文字幕| av高清资源| 国产在线观看91| 96精品视频| 狠狠操狠狠色| 亚洲综合色视频在线观看| 欧美xxxx黑人又粗又长| 国产三级做爰在线观看| 日本黄在线观看| 国产h色视频在线观看| 欧美成人精品福利网站| 国产成a人亚洲精v品| 国产尤物视频| xxx国产精品| 精品999视频| 99视频免费在线观看| 天天操夜夜做| 国产h色视频在线观看| 成年网站免费入口在线观看| av网站大全在线观看| 国产福利视频在线观看| 高清av中文在线字幕观看1| 一本久久精品| 精品福利视频导航大全| 国产青草视频在线观看视频| 国产在线看片| 毛片视频免费观看| 免费特级黄毛片| 欧美人成在线观看网站高清| 日本三级视频网站| 国产porny蝌蚪视频| 国产黄色在线播放| 国产精品久久久久久精| 99视频免费在线观看| 中文字幕日本三级| 天天操天天艹| 国产精品入口麻豆高清| 国产性网软件大全| 国产一级免费在线观看| 国产精品久久久精品a级小说| 国产欧美日韩第一页| 国产日本在线| 在线观看免费黄色| av在线免费观看网| 国产永久免费高清在线观看| 狠狠干在线视频| 最新黄网在线观看| 国产成+人+亚洲+欧美+综合| 中文资源在线官网| 开心快乐六月丁香婷婷| 国产一级二级在线| 在线一区观看| 国产美女福利在线| 6699久久国产精品免费| 欧美日韩在线精品成人综合网| 黄色av电影在线播放| 国产黄色在线网站| 欧美日韩在线资源| 精品99又大又爽又硬少妇毛片| 国产福利片在线| 自拍av在线| 国产精品入口麻豆电影| 国产经典av| 亚洲精品在线播放视频| www在线视频观看| 国产网站在线免费观看| 国产经典自拍视频在线观看| 精品国内自产拍在线视频| 在线视频1区2区| 国产麻豆一区二区三区精品 | 国产精品剧情一区二区三区| av在线1区2区| 国产高清大尺度一区二区不卡| av网址在线免费观看| 激情六月丁香| 午夜在线不卡| 51成人精品网站| 国产精品视频一区二区久久 | 精品街拍一区二区| 国产免费永久在线观看| h网站久久久| 尤物视频在线观看视频| 国产黄色片在线观看| 国产欧美日韩第一页| 中文字幕在线观看播放| 国产精品自产拍在线观看2019| 免费影视观看网站入口| 2019中文字幕在线电影免费| 中文字幕在线视频网| 狠狠色综合久久婷婷| 免费av在线| 久久精品最新免费国产成人| 不卡av免费观看| sm国产在线调教视频| 在线a人片免费观看视频| 精品一区二区91| 在线视频观看亚洲| av中文资源在线| 欧美专区日韩| 国产精品一区牛牛影视| 天堂资源最新版在线视频观看免费网 | av人人综合网| 日本亚洲精品| 日本天堂影院在线视频| 日本视频三区| 天天操天天操天天色天天要| 天天操天天艹| 在线伊人免费视频| 另类高清dbsm日本tvav| 国产视频第一区| 在线三级中文| 免费a级毛片在线观看| 国产免费av在线| 99色在线观看| 国产又色又爽又黄刺激在线视频| 丁香花高清在线观看完整版| 中文字幕不卡| 中文字幕av高清| 狠狠操视频网| 精品亚洲成a人片在线观看| 欧美日韩性视频一区二区三区| 黄色一级视频网站| 国产高清av| 日本中文字幕视频在线| av网址在线看| 国产三级视频在线看|