php钩子原理和实现
2017年3月18日17:22:52
php版本 5.6.27
5.3以下和5.3以上的版本在PHP类与对象区别很大,请注意
其实原理很简单,有些人把事情弄的过于发杂,其实就是调用某个目录下的比如/hook目录下注册在hook函数里面和读取hook配置文件里面的类的方法的一个调用类的方法的功能
目的就是最少改动代码,改动旧功能,或者增加一些新功能,或者简单说成调用函数都行
但是读取hook的配置文件,还是需要在系统的里面每次都需要读取,其实就失去了hook的意义,建议只做钩子本身的就好
参看一下ci的hook,仅截取hook函数核心部分
<?php
protected function _run_hook($data) {
// Closures/lambda functions and array($object, 'method') callables
if (is_callable($data)) {
is_array($data) ? $data[0]->{$data[1]}() : $data(); return TRUE;
} elseif (!is_array($data)) {
return FALSE;
} if ($this->_in_progress === TRUE) {
return;
} if (!isset($data['filepath'], $data['filename'])) {
return FALSE;
} $filepath = APPPATH . $data['filepath'] . '/' . $data['filename']; if (!file_exists($filepath)) {
return FALSE;
}
$class = empty($data['class']) ? FALSE : $data['class'];
$function = empty($data['function']) ? FALSE : $data['function'];
$params = isset($data['params']) ? $data['params'] : ''; if (empty($function)) {
return FALSE;
} // Set the _in_progress flag
$this->_in_progress = TRUE; // Call the requested class and/or function
if ($class !== FALSE) {
// The object is stored?
if (isset($this->_objects[$class])) {
if (method_exists($this->_objects[$class], $function)) {
$this->_objects[$class]->$function($params);
} else {
return $this->_in_progress = FALSE;
}
} else {
class_exists($class, FALSE) OR require_once($filepath); if (!class_exists($class, FALSE) OR ! method_exists($class, $function)) {
return $this->_in_progress = FALSE;
} // Store the object and execute the method
$this->_objects[$class] = new $class();
$this->_objects[$class]->$function($params);
// 核心部分 读取参数部分,去实例化类调用方法 传递参数 其实这也是MVC url路由实现的核心,现在很多
//都是使用 call_user_func_array call_user_func 这两个方法
}
} else {
function_exists($function) OR require_once($filepath); if (!function_exists($function)) {
return $this->_in_progress = FALSE;
} $function($params);
} $this->_in_progress = FALSE;
return TRUE;
}
原理图解

个人实现版本
如果你觉得麻烦,甚至可以写个方法都行,建议写成一个类,因为有些东西需要更多的信息

1.php
include 'hook.class.php'; $rr = new hook();
//$ee = $rr->get_all_class(); $rr->run_hook('ff','ss',array());
//echo '<pre>';
//print_r($ee);
//echo '</pre>';
hook.class.php
class hook {
public $HOOK_PATH;
public $PATH; //完整钩子文件目录
public $object;
//调用的时候这个类使用的时候,必须在系统的执行流程当中
public function __construct() {
$this->HOOK_PATH = ''; //项目的路径,根据你的项目计算路径
$current_path = str_replace("\\", "/", getcwd()); //获取当前目录
//这个地方在实际用的时候看你需要调整
$this->PATH = $current_path . $this->HOOK_PATH;
}
/* 注册钩子 也可以叫做运行钩子
* $class 类名称
* $function 方法
* $param 方法参数
*/
public function run_hook($class, $function, $param = array()) {
include $this->PATH . '/' . $class . '.class.php';
// var_dump($this->PATH . '/' . $class . '.class.php');
// call_user_func_array(array($class, $function), $param);//只能调用类的静态方法
// call_user_func(array($class, $function), $param); //只能调用类的静态方法
// 其他写法
$this->object = new $class();
$this->object->$function($param); //这样就可以不用调用静态方法
}
//返回当前已经所有的钩子类和方法 不要当前方法调用这个核心类,需要稍微改造,在$hook_array[$key]['function']的返回方法名的时候
public function get_all_class() {
//搜寻hook目录下的所有钩子文件,返回数组
// $this->PATH
// var_dump($this->PATH);
$file_array = scandir($this->PATH);
$hook_array = array();
foreach ($file_array as $key => $value) {
if (strpos($value, '.class.php') == true) { //扫描路径绝对不能和这个类本身在一个同一个目录下,不然会出现重复声明的同名类
$name = explode('.', $value);
$hook_array[$key]['name'] = $name['0'] . '钩子类';
$hook_array[$key]['url'] = $this->PATH . '/' . $value;
// include $hook_array[$key]['url'];
// $cc = new $name['0']();
// $hook_array[$key]['function'][] = get_class_methods($cc);
// $hook_array[$key]['function']['param'][] = get_class_vars($class_name); //获取方法变量
}
}
return $hook_array;
}
}
调用的某个类名
ff.class.php 的ss方法
public function ss() {
// static public function ss() {
echo 'dddddddddddddddddddd';
}
另一个版本
更方便调用
class hooks {
const Directory_Structure = '/hooks/'; //相对目录的路径 具体项目使用的时候需要调整
static public function get_path() {
return str_replace("\\", "/", getcwd());
}
static public function run_hook($class, $function, $param = array()) {
$s = include self::get_path() . self::Directory_Structure .$class. '.class.php';
call_user_func(array($class, $function), $param); //只能调用类的静态方法
// 其他写法
// $object = new $class();
// $object->$function($param); //这样就可以不用调用静态方法
}
}
使用
include 'hooks.class.php';
hooks::run_hook('ee', 'vv',$param =array());
当然也可以这么访问
$foo = new hooks();
$foo->run_hook('ee', 'vv',array());
$foo::run_hook('ee', 'vv',array());
自 PHP 5.3.0 起,可以用一个变量来动态调用类。但该变量的值不能为关键字 self,parent 或 static。
钩子是比较灵活的,可以额外增加一个功能代码,使代码更整洁,比如在做什么一些重要操作,创建订单,在创建订单之前需要做些什么,在创建之后做些什么,都可以使用钩子
这样代码更加灵活
php钩子原理和实现的更多相关文章
- Wordpress解析系列之PHP编写hook钩子原理简单实例
Wordpress作为全球应用最广泛的个人博客建站工具,有很多的技术架构值得我们学习推敲.其中,最著名最经典的编码技术架构就是采用了hook的机制. hook翻译成中文是钩子的意思,单独看这个词我们难 ...
- WordPress 插件机制的简单用法和原理(Hook 钩子)
WordPress 的插件机制实际上只的就是这个 Hook 了,它中文被翻译成钩子,允许你参与 WordPress 核心的运行,是一个非常棒的东西,下面我们来详细了解一下它. PS:本文只是简单的总结 ...
- 黄聪:WordPress动作钩子函数add_action()、do_action()源码解析
WordPress常用两种钩子,过滤钩子和动作钩子.过滤钩子相关函数及源码分析在上篇文章中完成,本篇主要分析动作钩子源码. 然而,在了解了动作钩子的源码后你会发现,动作钩子核心代码竟然跟过滤钩子差不多 ...
- WordPress中函数钩子hook的作用及基本用法
WordPress 的插件机制实际上只的就是这个 Hook 了,它中文被翻译成钩子,允许你参与 WordPress 核心的运行,是一个非常棒的东西,下面我们来详细了解一下它.钩子分类 钩子分为两种,一 ...
- git自定义项目钩子和全局钩子
钩子介绍 自定义钩子分为:项目钩子和全局钩子 自定义全局钩子: 全局钩子目录结构: (注意:excludes目录结构是我们自定义的目录,规则逻辑在update.d/update.py脚本里实现的,非g ...
- 使用python编写svn钩子
同上一篇trac中安装插件的文章的出发点一样,感觉用文档和口头制定规则在执行上会有偏差并且需要经常引导新人去熟悉规则. 所以,又费了几个小时去琢磨怎么改进svn提交代码的钩子,现有的钩子的功能比较简单 ...
- WordPress系列之钩子hook的作用及基本用法
WordPress 的插件机制实际上只的就是这个 Hook 了,它中文被翻译成钩子,允许你参与 WordPress 核心的运行,是一个非常棒的东西,下面我们来详细了解一下它.钩子分类 钩子分为两种,一 ...
- EPOLL内核原理极简图文解读(转)
预备知识:内核poll钩子原理内核函数poll_wait把当前进程加入到驱动里自定义的等待队列上 当驱动事件就绪后,就可以在驱动里自定义的等待队列上唤醒调用poll的进程 故poll_wait作用:可 ...
- python win32api 使用小技巧
前些日子,由于需要,用python写了个小插件,通过win32api 访问外部程序的窗口 并且做些小操作. 因为原来对win32api 不怎么熟悉 所以只好求救.群里有个QQ:32034767 唐骁勇 ...
随机推荐
- grid - 隐式地命名网格区域名称
通常可以将网格线命名成任何你想命名的名称,如果网格线名称添加-start和-end的后缀,其实也隐式的创建一个网格区域,可以用来设置网格项目的位置. 在这个示例中,行和列都具有inner-start和 ...
- Log4j/Log4j2自定义Appender来实现日志级别计数统计及监控
一.简述 本文主要讲如何基于Log4j2来实现自定义的Appender.一般用途是用于Log4j2自带的Appender不足以满足我们的需求,或者需要我们对日志进行拦截统计等操作时,需要我们自定义Ap ...
- CodeForces - 344E Read Time (模拟题 + 二分法)
E. Read Time time limit per test 1 second memory limit per test 256 megabytes input standard input o ...
- (整理)plsql导入dmp文件
转载请注明出处: https://www.cnblogs.com/darkknightzh/p/10093063.html 参考网址: https://www.2cto.com/database/20 ...
- 使用h2数据库
h2数据库提供了一个简单的web管理界面 import org.h2.tools.Server; import org.slf4j.Logger; import org.slf4j.LoggerFac ...
- linux内存管理之malloc、vmalloc、kmalloc的区别
kmalloc kzalloc vmalloc malloc 和get_free_page()的区别 一.简述 1. kmalloc申请的是较小的连续的物理内存,虚拟地址上也是连续的.kmalloc和 ...
- STM32F105 PA9/OTG_FS_VBUS Issues
https://www.cnblogs.com/shangdawei/p/3264724.html F105 DFU模式下PA9引脚用来检测USB线缆,若电平在2.7~5v则认为插入usb设备(检测到 ...
- MySql 建表、添加字段、修改字段、添加索引SQL语句写法及SQL索引
---------添加索引方法--------- .添加PRIMARY KEY(主键索引) mysql>ALTER TABLE `table_name` ADD PRIMARY KEY ( `c ...
- 关于python协程中aiorwlock 使用问题
最近工作中多个项目都开始用asyncio aiohttp aiomysql aioredis ,其实也是更好的用python的协程,但是使用的过程中也是遇到了很多问题,最近遇到的就是 关于aiorwl ...
- repo命令详解
Android 为企业提供一个新的市场,无论大企业,小企业都是处于同一个起跑线上.研究 Android 尤其是 Android 系统核心或者是驱动的开发,首先需要做的就是本地克隆建立一套 Androi ...