對"鉤子"這個(gè)概念其實(shí)不熟悉,最近看到一個(gè)php框架中用到這種機(jī)制來擴(kuò)展項(xiàng)目,所以大概來了解下。
所謂Hook機(jī)制,是從Windows編程中流行開的一種技術(shù)。其主要思想是提前在可能增加功能的地方埋好(預(yù)設(shè))一個(gè)鉤子,這個(gè)鉤子并沒有實(shí)際的意義,當(dāng)我們需要重新修改或者增加這個(gè)地方的邏輯的時(shí)候,把擴(kuò)展的類或者方法掛載到這個(gè)點(diǎn)即可。
hook插件機(jī)制的基本思想:
在項(xiàng)目代碼中,你認(rèn)為要擴(kuò)展(暫時(shí)不擴(kuò)展)的地方放置一個(gè)鉤子函數(shù),等需要擴(kuò)展的時(shí)候,把需要實(shí)現(xiàn)的類和函數(shù)掛載到這個(gè)鉤子上,就可以實(shí)現(xiàn)擴(kuò)展了。
思想就是這樣聽起來比較籠統(tǒng),看一個(gè)網(wǎng)上的實(shí)現(xiàn)的例子。
整個(gè)插件機(jī)制包含三個(gè)部分:
1.hook插件經(jīng)理類:這個(gè)是核心文件,是一個(gè)應(yīng)用程序全局Global對象。它主要有三個(gè)職責(zé)
1>監(jiān)聽已經(jīng)注冊了的所有插件,并實(shí)例化這些插件對象。
2>注冊所有插件。
3>當(dāng)鉤子條件滿足時(shí),觸發(fā)對應(yīng)的對象方法。
2.插件的功能實(shí)現(xiàn):這大多由第三方開發(fā)人員完成,但需要遵循我們(經(jīng)理類定義)的規(guī)則,這個(gè)規(guī)則是插件機(jī)制所規(guī)定的,因插件機(jī)制的不同而不同。
3.插件的觸發(fā):也就是鉤子的觸發(fā)條件。這是一小段代碼,放置在你需要調(diào)用插件的地方,用于觸發(fā)這個(gè)鉤子。
----------------------------------看一看別人實(shí)現(xiàn)的方案--------------------------------
首先是插件經(jīng)理類PluginManager,這個(gè)類要放在全局引用里面,在所有需要用到插件的地方,優(yōu)先加載。
?php
/**
*
* 插件機(jī)制的實(shí)現(xiàn)核心類
*/
class PluginManager
{
/**
* 監(jiān)聽已注冊的插件
*
* @access private
* @var array
*/
private $_listeners = array();
/**
* 構(gòu)造函數(shù)
*
* @access public
* @return void
*/
public function __construct()
{
#這里$plugin數(shù)組包含我們獲取已經(jīng)由用戶激活的插件信息
#為演示方便,我們假定$plugin中至少包含
#$plugin = array(
# 'name' => '插件名稱',
# 'directory'=>'插件安裝目錄'
#);
$plugins = get_active_plugins();#這個(gè)函數(shù)請自行實(shí)現(xiàn)
if($plugins)
{
foreach($plugins as $plugin)
{//假定每個(gè)插件文件夾中包含一個(gè)actions.php文件,它是插件的具體實(shí)現(xiàn)
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);
}
}
}
}
#此處做些日志記錄方面的東西
}
/**
* 注冊需要監(jiān)聽的插件方法(鉤子)
*
* @param string $hook
* @param object $reference
* @param string $method
*/
function register($hook, $reference, $method)
{
//獲取插件要實(shí)現(xiàn)的方法
$key = get_class($reference).'->'.$method;
//將插件的引用連同方法push進(jìn)監(jiān)聽數(shù)組中
$this->_listeners[$hook][$key] = array($reference, $method);
#此處做些日志記錄方面的東西
}
/**
* 觸發(fā)一個(gè)鉤子
*
* @param string $hook 鉤子的名稱
* @param mixed $data 鉤子的入?yún)?
* @return mixed
*/
function trigger($hook, $data='')
{
$result = '';
//查看要實(shí)現(xiàn)的鉤子,是否在監(jiān)聽數(shù)組之中
if (isset($this->_listeners[$hook]) is_array($this->_listeners[$hook]) count($this->_listeners[$hook]) > 0)
{
// 循環(huán)調(diào)用開始
foreach ($this->_listeners[$hook] as $listener)
{
// 取出插件對象的引用和方法
$class = $listener[0];
$method = $listener[1];
if(method_exists($class,$method))
{
// 動(dòng)態(tài)調(diào)用插件的方法
$result .= $class->$method($data);
}
}
}
#此處做些日志記錄方面的東西
return $result;
}
}
接下來是一個(gè)簡單插件的實(shí)現(xiàn)DEMO_actions。這是一個(gè)簡單的Hello World插件,用于輸出一句話。在實(shí)際情況中,say_hello可能包括對數(shù)據(jù)庫的操作,或者是其他一些特定的邏輯。
?php
/**
* 這是一個(gè)Hello World簡單插件的實(shí)現(xiàn)
*/
/**
*需要注意的幾個(gè)默認(rèn)規(guī)則:
* 1. 本插件類的文件名必須是action
* 2. 插件類的名稱必須是{插件名_actions}
*/
class DEMO_actions
{
//解析函數(shù)的參數(shù)是pluginManager的引用
function __construct($pluginManager)
{
//注冊這個(gè)插件
//第一個(gè)參數(shù)是鉤子的名稱
//第二個(gè)參數(shù)是pluginManager的引用
//第三個(gè)是插件所執(zhí)行的方法
$pluginManager->register('demo', $this, 'say_hello');
}
function say_hello()
{
echo 'Hello World';
}
}
再接下來就是插件的調(diào)用觸發(fā)的地方,比如我要將say_hello放到我博客首頁Index.php, 那么你在index.php中的某個(gè)位置寫下:
$pluginManager->trigger('demo','');
第一個(gè)參數(shù)表示鉤子的名字,第二個(gè)參數(shù)是插件對應(yīng)方法的入口參數(shù),由于這個(gè)例子中沒有輸入?yún)?shù),所以為空。
這樣一個(gè)例子基本上很明確的表達(dá)了"鉤子"插件機(jī)制的實(shí)現(xiàn)方式和邏輯。
以上就是本文的全部內(nèi)容,希望對大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
您可能感興趣的文章:- thinkPHP中鉤子的兩種配置調(diào)用方法詳解
- thinkPHP中鉤子的使用方法實(shí)例分析
- PHP鉤子與簡單分發(fā)方式實(shí)例分析
- thinkPHP基于反射實(shí)現(xiàn)鉤子的方法分析
- PHP鉤子實(shí)現(xiàn)方法解析
- php中的鉤子理解及應(yīng)用實(shí)例分析
- CI框架源碼解讀之利用Hook.php文件完成功能擴(kuò)展的方法
- php面向?qū)ο笾衧tatic靜態(tài)屬性和靜態(tài)方法的調(diào)用
- PHP 面向?qū)ο?final類與final方法
- PHP面向?qū)ο笕筇攸c(diǎn)學(xué)習(xí)(充分理解抽象、封裝、繼承、多態(tài))
- PHP對象Object的概念 介紹
- php中鉤子(hook)的原理與簡單應(yīng)用demo示例