php钩子原理和实现
2017年3月18日17:22:52?
php版本 5.6.27
5.3以下和5.3以上的版本在PHP類與對象區(qū)別很大,請注意
?
其實原理很簡單,有些人把事情弄的過于發(fā)雜,其實就是調(diào)用某個目錄下的比如/hook目錄下注冊在hook函數(shù)里面和讀取hook配置文件里面的類的方法的一個調(diào)用類的方法的功能
目的就是最少改動代碼,改動舊功能,或者增加一些新功能,或者簡單說成調(diào)用函數(shù)都行
?
但是讀取hook的配置文件,還是需要在系統(tǒng)的里面每次都需要讀取,其實就失去了hook的意義,建議只做鉤子本身的就好
?參看一下ci的hook,僅截取hook函數(shù)核心部分
<?phpprotected function _run_hook($data) {// Closures/lambda functions and array($object, 'method') callablesif (is_callable($data)) {is_array($data) ? $data[0]->{$data[1]}() : $data();return TRUE;} elseif (!is_array($data)) {return FALSE;}if ($this->_in_progress === TRUE) {return;}if (!isset($data['filepath'], $data['filename'])) {return FALSE;}$filepath = APPPATH . $data['filepath'] . '/' . $data['filename'];if (!file_exists($filepath)) {return FALSE;}$class = empty($data['class']) ? FALSE : $data['class'];$function = empty($data['function']) ? FALSE : $data['function'];$params = isset($data['params']) ? $data['params'] : '';if (empty($function)) {return FALSE;}// Set the _in_progress flag$this->_in_progress = TRUE;// Call the requested class and/or functionif ($class !== FALSE) {// The object is stored?if (isset($this->_objects[$class])) {if (method_exists($this->_objects[$class], $function)) {$this->_objects[$class]->$function($params);} else {return $this->_in_progress = FALSE;}} else {class_exists($class, FALSE) OR require_once($filepath);if (!class_exists($class, FALSE) OR ! method_exists($class, $function)) {return $this->_in_progress = FALSE;}// Store the object and execute the method$this->_objects[$class] = new $class(); $this->_objects[$class]->$function($params);// 核心部分 讀取參數(shù)部分,去實例化類調(diào)用方法 傳遞參數(shù) 其實這也是MVC url路由實現(xiàn)的核心,現(xiàn)在很多
//都是使用 call_user_func_array call_user_func 這兩個方法
}} else {function_exists($function) OR require_once($filepath);if (!function_exists($function)) {return $this->_in_progress = FALSE;}$function($params);}$this->_in_progress = FALSE;return TRUE;}
?
原理圖解
?
個人實現(xiàn)版本
?如果你覺得麻煩,甚至可以寫個方法都行,建議寫成一個類,因為有些東西需要更多的信息
?
1.php
include 'hook.class.php';$rr = new hook(); //$ee = $rr->get_all_class();$rr->run_hook('ff','ss',array()); //echo '<pre>'; //print_r($ee); //echo '</pre>';?
?
hook.class.php
class hook {public $HOOK_PATH;public $PATH; //完整鉤子文件目錄public $object;//調(diào)用的時候這個類使用的時候,必須在系統(tǒng)的執(zhí)行流程當(dāng)中public function __construct() {$this->HOOK_PATH = ''; //項目的路徑,根據(jù)你的項目計算路徑 $current_path = str_replace("\\", "/", getcwd()); //獲取當(dāng)前目錄//這個地方在實際用的時候看你需要調(diào)整$this->PATH = $current_path . $this->HOOK_PATH;}/* 注冊鉤子 也可以叫做運行鉤子* $class 類名稱* $function 方法 * $param 方法參數(shù)*/public function run_hook($class, $function, $param = array()) {include $this->PATH . '/' . $class . '.class.php'; // var_dump($this->PATH . '/' . $class . '.class.php'); // call_user_func_array(array($class, $function), $param);//只能調(diào)用類的靜態(tài)方法 // call_user_func(array($class, $function), $param); //只能調(diào)用類的靜態(tài)方法 // 其他寫法$this->object = new $class();$this->object->$function($param); //這樣就可以不用調(diào)用靜態(tài)方法 }//返回當(dāng)前已經(jīng)所有的鉤子類和方法 不要當(dāng)前方法調(diào)用這個核心類,需要稍微改造,在$hook_array[$key]['function']的返回方法名的時候public function get_all_class() {//搜尋hook目錄下的所有鉤子文件,返回數(shù)組 // $this->PATH // var_dump($this->PATH);$file_array = scandir($this->PATH);$hook_array = array();foreach ($file_array as $key => $value) {if (strpos($value, '.class.php') == true) { //掃描路徑絕對不能和這個類本身在一個同一個目錄下,不然會出現(xiàn)重復(fù)聲明的同名類$name = explode('.', $value);$hook_array[$key]['name'] = $name['0'] . '鉤子類';$hook_array[$key]['url'] = $this->PATH . '/' . $value; // include $hook_array[$key]['url']; // $cc = new $name['0'](); // $hook_array[$key]['function'][] = get_class_methods($cc);// $hook_array[$key]['function']['param'][] = get_class_vars($class_name); //獲取方法變量
}}return $hook_array;}}
?
?
調(diào)用的某個類名
ff.class.php 的ss方法
?
public function ss() { // static public function ss() {echo 'dddddddddddddddddddd';}?
另一個版本
更方便調(diào)用
class hooks {const Directory_Structure = '/hooks/'; //相對目錄的路徑 具體項目使用的時候需要調(diào)整static public function get_path() {return str_replace("\\", "/", getcwd());}static public function run_hook($class, $function, $param = array()) {$s = include self::get_path() . self::Directory_Structure .$class. '.class.php';call_user_func(array($class, $function), $param); //只能調(diào)用類的靜態(tài)方法 // 其他寫法 // $object = new $class(); // $object->$function($param); //這樣就可以不用調(diào)用靜態(tài)方法 }}?
使用
include 'hooks.class.php';hooks::run_hook('ee', 'vv',$param =array());當(dāng)然也可以這么訪問
$foo = new hooks();
$foo->run_hook('ee', 'vv',array());
$foo::run_hook('ee', 'vv',array());
自 PHP 5.3.0 起,可以用一個變量來動態(tài)調(diào)用類。但該變量的值不能為關(guān)鍵字?self,parent?或?static。
?鉤子是比較靈活的,可以額外增加一個功能代碼,使代碼更整潔,比如在做什么一些重要操作,創(chuàng)建訂單,在創(chuàng)建訂單之前需要做些什么,在創(chuàng)建之后做些什么,都可以使用鉤子
這樣代碼更加靈活
轉(zhuǎn)載于:https://www.cnblogs.com/zx-admin/p/6564649.html
總結(jié)
以上是生活随笔為你收集整理的php钩子原理和实现的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: Android -- 自定义StepVi
- 下一篇: JavaScript中innerHTML