今天研究了一下ThinkPHP的钩子,也算是框架的核心功能之一了,感觉很不错。主要的两个方向就是行为扩展和插件,这两个都是业务逻辑比较独立的,无需改动框架即可实现某个功能。
行为(Behavior),以下从tp3.2手册摘了一段,还是比较好理解的。
你可以想象成在应用执行过程中的一个动作或者处理,在框架的执行流程中,各个位置都可以有行为产生,例如路由检测是一个行为,静态缓存是一个行为,用户权限检测也是行为,大到业务逻辑,小到浏览器检测、多语言检测等等都可以当做是一个行为,甚至说你希望给你的网站用户的第一次访问弹出Hello,world!这些都可以看成是一种行为,行为的存在让你无需改动框架和应用,而在外围通过扩展或者配置来改变或者增加一些功能。
你用来标记某个行为的名称,我们就称之为标签(tag),tp框架里有内置的系统标签,如app_init、app_begin等,这些标签就标记某些位置的行为(处理该行为)。
行为标签的定义,一般是在配置文件中 Common/conf/tags.php
return array(
'mytest' => array('Portal\\Behavior\\testBehavior'),
);
这里前面就是钩子,而后面的array 是执行为处理的位置,可以对应多个,是按顺序依次执行。这是自动钩子定义,还可以手动定义标签
3.2.1版本之前
\Think\Hook::add('action_begin','Home\\Behaviors\\test1Behavior');
3.2.1版本之后
\Think\Hook::add('action_begin','Home\\Behaviors\\test1Behavior');
两个方法都是可以的,这里需要注意一下,现在很多网上关于钩子和行为的博客解说的都是3.2.1之前的,是按照插件的路径去找执行方法的。
不得不说一下,tp这里的处理比较坑。。
static public function exec($name, $tag,&$params=NULL) {
if('Behavior' == substr($name,-8) ){
// 行为扩展必须用run入口方法
$class = $name;
$tag = 'run';
}else{
$class = "plugins\\{$name}\\{$name}Plugin";
}
if(class_exists($class)){ //ThinkCMF NOTE 插件或者行为存在时才执行
$addon = new $class();
return $addon->$tag($params);
}
}
这里检测了class不存在之后啥都没有,debug下不记录日志也没trace,也是醉了。
上面的代码写的很清楚,$name是指tag的执行路径,如果name后缀为Behavior的就作为行为扩展执行,如果不是,则作为plugns下的插件执行。
- 行为的触发
两种方式,看代码就知道其实就是tag调用Hook:listen
\Think\Hook::listen('标签名'[,参数]);
tag('标签名'[,参数]);
- 行为扩展
行为扩展的定义,run在父类中是抽象方法所以必须重写run方法,而行为扩展的入口就是run方法,有且只有一个run。
<?php
namespace Home\Behaviors;
class testBehavior extends \Think\Behavior{
//行为执行入口
public function run(&$param){
}
}
- 插件
插件的定义,在plugins目录下,插件名作为文件夹test,下面有testPlugin.class.php,这样的结构才能被tp找到,这里只是一个简单的主体Controller,test插件文件夹可以根据情况配置Controller、model、view等模块,简直不要太强大。
<?php
namespace plugins\test;
use Common\Lib\Plugin;
class testPlugin extends Plugin {
public function mytest($param){
echo 'this is plugin test-mytest';
}
public function yourtest($param){
echo 'this is plugin test-yourtest'; //另一个插件钩子的执行入口
}
}
和行为扩展不同,一个插件中可以包含多个方法,作为不同钩子的的处理方法,这应该是最大的区别。
**粗体** _斜体_ [链接](http://example.com) `代码` - 列表 > 引用
。你还可以使用@
来通知其他用户。