序
作为程序员,设计出优雅而完美的系统,永远是让我们非常兴奋的事情。高手不在于你会多少语言,而在于你有多高的思想。
在设计中,怎么体现自身价值,那就是要比别人多想几步。
讲钩子程序,起源是对用户提交的参数校验(永远不要相信用户),一开始为了赶工期,按照比较传统的方式,每个接口里重复性的对参数进行过滤。后面随着业务的发展(功能迭代),系统的维护成本越来越高,遂想一个更高级的方式进行处理。借鉴同事之前的代码,使用钩子方式进行重构。
之前写过javascript 钩子机制, 偏后钩,可以互相借鉴下。
脉路
概念
把一段程序块(执行体)通过某种方式挂入系统中,从而获得对系统的控制权。
注意下图挂钩位置:
应用
小的方面: 进行基础的入参校验或消息过滤。
大的方面:组件化,可在系统中进行插拔管理。
优点:
1、降低系统的耦合度;
2、降低开发、测试人力成本,用少量的代码实现高可用功能;
3、提高模块间的可用性;
4、通过配置(配置文件or数据库)的方式升级接口。
缺点:
学习成本过高;
系统复杂度提升;
实现思想
配置文件的方式进行钩子定义、钩子链管理(使用“组”的概念)、挂钩。
钩子:程序执行体;
钩子组: 钩子链的分类定义;
挂钩: 入口(MVC中action或者controller)与钩子组进行绑定。
实现方式
挂钩器(继承类):
<?php
/**
* @name Service_Page_Test
* @desc page层对接第三方抽象类
* @author
*/
abstract class Service_Page_Test
{
public $hookGroupPrev = null; // 前钩子组
public $hookGroupAfter = null; // 后钩子组
public $hookReturn = array(); //钩子返回值
public $reqData = null; // page模块分析的数据 /**
* 获取需要验证的参数配置
* @return array
*/
public function _getCheckParams()
{
return array();
} /**
* 入口方法
* @param array $arrInput
* @return array
*/
public function execute($arrInput)
{
$res = array(
'errno' => Test_Errno::ERRNO_SUCCESS,
'errmsg' => Test_Errno::$ERRMSG[Test_Errno::ERRNO_SUCCESS],
);
try {
$this->_init($arrInput);
$this->_beforeExecute();
$res = $this->doExecute($arrInput);
$this->_afterExecute();
} catch (Test_Exception $e) {
$res = array(
'errno' => $e->getCode(),
'errmsg' => $e->getMessage(),
);
} catch (Exception $e) {
$res = array(
'errno' => $e->getCode(),
'errmsg' => $e->getMessage(),
); }
return $res;
} /**
* auto exec
* @param array $arrInput
* @throws Exception
* @return array
*/
protected function doExecute($arrInput){
} /**
* 获取权限信息
* @param array $arrInput
* @return array
*/
public function _init($arrInput)
{
$pageModulesConf = Conf::getConf('page/' . get_class($this));
$this->reqData = $arrInput;
$this->hookGroupPrev[] = $pageModulesConf['hook_group']['prev'];
$this->hookGroupAfter[] = $pageModulesConf['hook_group']['after'];
} /**
* 执行filter
* @param string
*/
public function _beforeExecute()
{
if (!empty($this->hookGroupPrev) && is_array($this->hookGroupPrev)) {
foreach ($this->hookGroupPrev as $hookGroups) {
foreach ($hookGroups as $hookGroup) {
$this->_executeHook($hookGroup, $this->reqData);
}
}
}
} /**
* @param array $arrInput
* @return array
*/
public function _afterExecute()
{
if (!empty($this->hookGroupAfter) && is_array($this->hookGroupAfter)) {
foreach ($this->hookGroupAfter as $hookGroups) {
foreach ($hookGroups as $hookGroup) {
$this->_executeHook($hookGroup, $this->reqData);
}
}
}
} /**
* 执行filter
* @param string
*/
public function _executeHook($hookGroup, $reqData)
{ $hookGroupConf = Conf::getConf('hook/group/' . $hookGroup);
if(!empty($hookGroupConf)){
foreach($hookGroupConf as $hook){
$hookConf = Conf::getConf('hook/hook/' . $hook);
$class = $hookConf['class'];
$method = $hookConf['method'];
$inputParams = isset($hookConf['getInputParams']) ? $this->{$hookConf['getInputParams']}() : null;
if (class_exists($class)) {
$obj = new $class();
if (method_exists($obj, $method)) {
$this->hookReturn[$hook][] = $obj->$method($inputParams, $reqData);
}
}
}
} } }
hook.conf
# 钩子组
[group]
[.check_req_customer]
: checkReqCustomerBaseInfo
[.after_demo]
: afterDemo # 钩子
[hook]
[.checkReqCustomerBaseInfo]
class: Service_Page_Hook_Customer
method: checkBaseInfo
getInputParams: _getCheckParams [.afterDemo]
class: Service_Page_Hook_Customer
method: afterDemo
getInputParams: _getCheckParams
page.conf
[Service_Page_Input]
#绑定钩子组
[.hook_group]
[..prev]
0 : check_req_customer
[..after]
0 : after_demo
推荐