[TOC]
* * * * *
## 1 監聽回調注冊源代碼(thinkphp/library/think/Hook.php)
~~~
private static $tags = [];
~~~
~~~
public static function add($tag, $behavior, $first = false)
{
if (!isset(self::$tags[$tag])) {
self::$tags[$tag] = [];
}
if (is_array($behavior)) {
self::$tags[$tag] = array_merge(self::$tags[$tag], $behavior);
} elseif ($first) {
array_unshift(self::$tags[$tag], $behavior);
} else {
self::$tags[$tag][] = $behavior;
}
}
~~~
~~~
public static function import(array $tags, $recursive = true)
{
if (!$recursive) {
self::$tags = array_merge(self::$tags, $tags);
} else {
foreach ($tags as $tag => $val) {
if (!isset(self::$tags[$tag])) {
self::$tags[$tag] = [];
}
if (!empty($val['_overlay'])) {
unset($val['_overlay']);
self::$tags[$tag] = $val;
} else {
self::$tags[$tag] = array_merge(self::$tags[$tag], $val);
}
}
}
}
~~~
~~~
public static function get($tag = '')
{
if (empty($tag)) {
return self::$tags;
} else {
return self::$tags[$tag];
}
}
~~~
~~~
public static function listen($tag, &$params = null)
{
if (isset(self::$tags[$tag])) {
foreach (self::$tags[$tag] as $name) {
if (APP_DEBUG) {
Debug::remark('behavior_start', 'time');
}
$result = self::exec($name, $tag, $params);
if (APP_DEBUG) {
Debug::remark('behavior_end', 'time');
Log::record('[ BEHAVIOR ] Run ' . ($name instanceof \Closure ? 'Closure' : $name) . ' @' . $tag . ' [ RunTime:' . Debug::getRangeTime('behavior_start', 'behavior_end') . 's ]', 'info');
}
if (false === $result) {
return;
}
}
}
return;
}
~~~
~~~
public static function exec($class, $tag = '', &$params = null)
{
if ($class instanceof \Closure) {
return $class($params);
}
$obj = new $class();
return ($tag && is_callable([$obj, $tag])) ? $obj->$tag($params) : $obj->run($params);
}
~~~
## 2 分析
Hook.php是框架的監聽回調注冊實現文件。也就是事件注冊機制的另一種說法。
總的思路是在需要回調的地方插入$tags標簽,然后實現相關業務邏輯$behavior。
系統運行過程中,遇到$tags標簽,自動運行相關業務邏輯$behavior。
1 靜態變量數組$tags
緩存注冊的標簽行為關聯
`private static $tags = [];`
2 add()
注冊標簽$tags行為$behavior關聯.$first可以指定優先級
`public static function add($tag, $behavior, $first = false){}`
3import()
以數組形式注冊$tags與$behaviro。$recursive是否覆蓋注冊
~~~
public static function import(array $tags, $recursive = true){}
~~~
4 get()
獲取$tags對應行為信息。無參獲取全部標簽的行為信息
`public static function get($tag = '')`
5 listen()
注冊$tag標簽的監聽。系統運行過程遇到$tag,自動運行關聯$begavior
`public static function listen($tag, &$params = null)`
6 exec()
以類名方式運行標簽$tags對應的行為
`public static function exec($class, $tag = '', &$params = null){}`
~~~
if ($class instanceof \Closure) {
return $class($params);
}
~~~
這段代碼直接返回對應類的對象。而不是返回$tag標簽對應的方法
## 3 總結
Hook.php 實現了框架的監聽注冊機制。可以用來在原有業務邏輯中,加入新的業務邏輯,而無需大幅度修改系統結構。使用方法見 [事件與插件注冊](http://www.hmoore.net/zmwtp/tp5/120039)
- 更新記錄
- 概述
- 文件索引
- 函數索引
- 章節格式
- 框架流程
- 前:章節說明
- 主:(index.php)入口
- 主:(start.php)框架引導
- 主:(App.php)應用啟動
- 主:(App.php)應用調度
- C:(Controller.php)應用控制器
- M:(Model.php)數據模型
- V:(View.php)視圖對象
- 附:(App.php)應用啟動
- 附:(base.php)全局變量
- 附:(common.php)模式配置
- 附:(convention.php)全局配置
- 附:(Loader.php)自動加載器
- 附:(Build.php)自動生成
- 附:(Hook.php)監聽回調
- 附:(Route.php)全局路由
- 附:(Response.php)數據輸出
- 附:(Log.php)日志記錄
- 附:(Exception.php)異常處理
- 框架工具
- 另:(helper.php)輔助函數
- 另:(Cache.php)數據緩存
- 另:(Cookie.php)cookie操作
- 另:(Console.php)控制臺
- 另:(Debug.php)開發調試
- 另:(Error.php)錯誤處理
- 另:(Url.php)Url操作文件
- 另:(Loader.php)加載器實例化
- 另:(Input.php)數據輸入
- 另:(Lang.php)語言包管理
- 另:(ORM.php)ORM基類
- 另:(Process.php)進程管理
- 另:(Session.php)session操作
- 另:(Template.php)模板解析
- 框架驅動
- D:(\config)配置解析
- D:(\controller)控制器擴展
- D:(\model)模型擴展
- D:(\db)數據庫驅動
- D:(\view)模板解析
- D:(\template)模板標簽庫
- D:(\session)session驅動
- D:(\cache)緩存驅動
- D:(\console)控制臺
- D:(\process)進程擴展
- T:(\traits)Trait目錄
- D:(\exception)異常實現
- D:(\log)日志驅動
- 使用范例
- 服務器與框架的安裝
- 控制器操作
- 數據模型操作
- 視圖渲染控制
- MVC開發初探
- 模塊開發
- 入口文件定義全局變量
- 運行模式開發
- 框架配置
- 自動生成應用
- 事件與插件注冊
- 路由規則注冊
- 輸出控制
- 多種應用組織
- 綜合應用
- tp框架整合后臺auto架構快速開發
- 基礎原理
- php默認全局變量
- php的魔術方法
- php命名空間
- php的自動加載
- php的composer
- php的反射
- php的trait機制
- php設計模式
- php的系統時區
- php的異常錯誤
- php的輸出控制
- php的正則表達式
- php的閉包函數
- php的會話控制
- php的接口
- php的PDO
- php的字符串操作
- php的curl
- 框架心得
- 心:整體結構
- 心:配置詳解
- 心:加載器詳解
- 心:輸入輸出詳解
- 心:url路由詳解
- 心:模板詳解
- 心:模型詳解
- 心:日志詳解
- 心:緩存詳解
- 心:控制臺詳解
- 框架更新
- 4.20(驗證類,助手函數)
- 4.27(新模型Model功能)
- 5.4(新數據庫驅動)
- 7.28(自動加載)