對"鉤子"這個概念其實不熟悉,最近看到一個php框架中用到這種機制來擴展項目,因此大概來了解下。php
hook插件機制的基本思想:html
在項目代碼中,你認爲要擴展(暫時不擴展)的地方放置一個鉤子函數,等須要擴展的時候,把須要實現的類和函數掛載到這個鉤子上,就能夠實現擴展了。數據庫
思想就是這樣聽起來比較籠統,看一個網上的實現的例子。數組
整個插件機制包含三個部分:php框架
1.hook插件經理類:這個是核心文件,是一個應用程序全局Global對象。它主要有三個職責框架
1>監聽已經註冊了的全部插件,並實例化這些插件對象。函數
2>註冊全部插件。this
3>當鉤子條件知足時,觸發對應的對象方法。spa
2.插件的功能實現:這大多由第三方開發人員完成,但須要遵循咱們(經理類定義)的規則,這個規則是插件機制所規定的,因插件機制的不一樣而不一樣。插件
3.插件的觸發:也就是鉤子的觸發條件。這是一小段代碼,放置在你須要調用插件的地方,用於觸發這個鉤子。
----------------------------------看一看別人實現的方案--------------------------------
首先是插件經理類PluginManager,這個類要放在全局引用裏面,在全部須要用到插件的地方,優先加載。
<?php /** * * 插件機制的實現核心類 */ class PluginManager { /** * 監聽已註冊的插件 * * @access private * @var array */ private $_listeners = array(); /** * 構造函數 * * @access public * @return void */ public function __construct() { #這裏$plugin數組包含咱們獲取已經由用戶激活的插件信息 #爲演示方便,咱們假定$plugin中至少包含 #$plugin = array( # 'name' => '插件名稱', # 'directory'=>'插件安裝目錄' #); $plugins = get_active_plugins();#這個函數請自行實現 if($plugins) { foreach($plugins as $plugin) {//假定每一個插件文件夾中包含一個actions.php文件,它是插件的具體實現 if (@file_exists(STPATH .'plugins/'.$plugin['directory'].'/actions.php')) { include_once(STPATH .'plugins/'.$plugin['directory'].'/actions.php'); $class = $plugin['name'].'_actions'; if (class_exists($class)) { //初始化全部插件 new $class($this); } } } } #此處作些日誌記錄方面的東西 } /** * 註冊須要監聽的插件方法(鉤子) * * @param string $hook * @param object $reference * @param string $method */ function register($hook, &$reference, $method) { //獲取插件要實現的方法 $key = get_class($reference).'->'.$method; //將插件的引用連同方法push進監聽數組中 $this->_listeners[$hook][$key] = array(&$reference, $method); #此處作些日誌記錄方面的東西 } /** * 觸發一個鉤子 * * @param string $hook 鉤子的名稱 * @param mixed $data 鉤子的入參 * @return mixed */ function trigger($hook, $data='') { $result = ''; //查看要實現的鉤子,是否在監聽數組之中 if (isset($this->_listeners[$hook]) && is_array($this->_listeners[$hook]) && count($this->_listeners[$hook]) > 0) { // 循環調用開始 foreach ($this->_listeners[$hook] as $listener) { // 取出插件對象的引用和方法 $class =& $listener[0]; $method = $listener[1]; if(method_exists($class,$method)) { // 動態調用插件的方法 $result .= $class->$method($data); } } } #此處作些日誌記錄方面的東西 return $result; } }
接下來是一個簡單插件的實現DEMO_actions。這是一個簡單的Hello World插件,用於輸出一句話。在實際狀況中,say_hello可能包括對數據庫的操做,或者是其餘一些特定的邏輯。
<?php /** * 這是一個Hello World簡單插件的實現 */ /** *須要注意的幾個默認規則: * 1. 本插件類的文件名必須是action * 2. 插件類的名稱必須是{插件名_actions} */ class DEMO_actions { //解析函數的參數是pluginManager的引用 function __construct(&$pluginManager) { //註冊這個插件 //第一個參數是鉤子的名稱 //第二個參數是pluginManager的引用 //第三個是插件所執行的方法 $pluginManager->register('demo', $this, 'say_hello'); } function say_hello() { echo 'Hello World'; } }
再接下來就是插件的調用觸發的地方,好比我要將say_hello放到我博客首頁Index.php, 那麼你在index.php中的某個位置寫下:
$pluginManager->trigger('demo','');
第一個參數表示鉤子的名字,第二個參數是插件對應方法的入口參數,因爲這個例子中沒有輸入參數,因此爲空。
這樣一個例子基本上很明確的表達了"鉤子"插件機制的實現方式和邏輯。