您的位置:首页 > 编程语言 > PHP开发

php钩子程序设计

2017-01-16 18:41 423 查看

 
  作为程序员,设计出优雅而完美的系统,永远是让我们非常兴奋的事情。高手不在于你会多少语言,而在于你有多高的思想。

  在设计中,怎么体现自身价值,那就是要比别人多想几步。

  讲钩子程序,起源是对用户提交的参数校验(永远不要相信用户),一开始为了赶工期,按照比较传统的方式,每个接口里重复性的对参数进行过滤。后面随着业务的发展(功能迭代),系统的维护成本越来越高,遂想一个更高级的方式进行处理。借鉴同事之前的代码,使用钩子方式进行重构。

  之前写过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]
0 : checkReqCustomerBaseInfo
[.after_demo]
0 : 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


  

 

推荐

内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: