一、简介
你可以把行为理解成是“在程序执行过程中的某一个位置会调起一个或一类事件”的动作。行为发生作用的位置我们称之为钩子,当应用程序运行到这个钩子的时候,就会被拦截下来,统一执行相关的行为。
类似于AOP编程中的“切面”的概念,给某一个钩子绑定相关行为就成了一种类AOP编程的思想。
一个完整的行为事件包括以下三项:
1)行为定义
2)行为绑定
3)监听钩子
ThinkPHP关于行为的核心方法都定义于核心文件thinkphp\library\think\Hook.php中。
二、行为定义
行为类一般放置于模块目录下的behavior目录里,当然 这不是硬性要求,你也可以按照你的喜好自定义目录。
行为类的定义很简单,一般来说只需要定义一个行为入口方法run()即可。如,我需要给我的后台管理系统做一个用户登录行为检测:
namespace app\admin\behavior; /** * Test: Use to learn ThinkPHP-Hook. */ class MyHook { public function run($params) { echo '自定义钩子的行为'; } }
行为的入口方法名称支持自定义,如果需要更改 在应用公共文件(common.php)中添加下面的代码即可:
Hook::portal('portal');
一个钩子可以注册多个行为,执行到某个钩子位置后,会按照注册的顺序依次执行相关的行为。但在某些特殊的情况下,你可以设置某个钩子只能执行一次行为,又或者你可以在一个钩子的某个行为中返回false来强制终止后续的行为执行;一个行为可以同时注册到多个不同的钩子上,完全看应用的需求来设计。
可以在行为方法中使用依赖注入,例如:
namespace app\index\behavior; use think\Request; class Test { public function run(Request $request, $params) { // 行为逻辑 } }
三、行为绑定
行为定义完成后,就需要绑定到某个标签位置(钩子)才能生效,否则是不会执行的。
3.1、通过配置文件
我们可以直接在应用目录下面或者模块的目录下面定义tags.php文件来统一定义行为,定义格式如下:
// 应用行为扩展定义文件return [ // 模块初始化 'module_init' => ['app\\admin\\behavior\\Login'], // 操作开始执行 'action_begin' => [], // 视图内容过滤 'view_filter' => [], // 日志写入 'log_write' => [], // 应用结束 'app_end' => [], //自定义钩子 => 绑定相应的行为 'my_hook' => ['app\\admin\\behavior\\MyHook'],];
3.2、使用think\facade\Hook类的add方法注册行为
Hook::add('my_hook','app\\admin\\behavior\\MyHook');
注意:
1)Hook::add要执行在Hook::listen之前,否则不会绑定成功。
2)Hook::add要么调用run方法要么调用当前钩子名称(驼峰法)的方法。
如果需要使用Hook::add调用其它方法,可以定义静态方法(app\index\behavior\CheckAuth::hello)或者使用闭包。
四、监听钩子
钩子的位置必须是事先设计好的,无论是框架还是应用的,要设置一个钩子,只需要在相关的位置添加一行代码(事先需要引入think\facade\Hook类),语法如下:
Hook::listen('钩子名称','参数','是否只有一次有效返回值');
4.1、系统钩子
系统钩子就是框架已经默认设置好的,开发者可以直接使用。
4.2、自定义钩子
Hook::listen('module_init'); //监听系统钩子 module_init Hook::listen('my_hook'); //监听自定义钩子 my_hook
五、闭包支持
可以不用定义行为直接把闭包函数绑定到某个标签位,例如:
Hook::add('hook_function',function($params){ var_dump($params); }); Hook::listen('hook_function','Hi nosee!');
六、直接执行行为
如果需要,你也可以不绑定行为标签,直接调用某个行为,例如:
//执行 app\index\behavior\CheckAuth行为类的run方法 并引用传入params参数 $result = Hook::exec('app\\index\\behavior\\CheckAuth', $params);
直接执行行为的时候,执行的是行为类中的run方法,如果需要执行行为类中的其它方法,可以使用:
//执行 app\index\behavior\CheckAuth行为类的hello方法 并引用传入params参数 $result = Hook::exec(['app\\index\\behavior\\CheckAuth','hello'], $params);
七、官方资料
官方手册地址:https://www.kancloud.cn/manual/thinkphp5_1/354129
八、尾声
在ThinkPHP6.0中,5.1中的行为已经升级成为事件了,也就说6.0中的事件系统可以看成是5.1版本行为系统的升级版,事件系统相比行为系统强大的地方在于事件本身可以是一个类,并且可以更好的支持事件订阅者。
声明:禁止任何非法用途使用,凡因违规使用而引起的任何法律纠纷,本站概不负责。
精彩评论