Wordpress解析系列之PHP编写hook钩子原理简单实例
Wordpress作为全球应用最广泛的个人博客建站工具,有很多的技术架构值得我们学习推敲。其中,最著名最经典的编码技术架构就是采用了hook的机制。
hook翻译成中文是钩子的意思,单独看这个词我们难以理解这个hook机制(即钩子机制)是什么意思。那么笔者就用大白话以通俗易懂方式给大家讲解一下什么是hook机制,以及用原生PHP函数编写实现简单实例。
大白话解释:以Wordpress为例,它的hook机制就是在网页加载时一起加载了很多hook变量,也就是钩子变量,这些变量作用是绑定相关的函数,只要hook变量被加载,Wordpress就会用一个内置通用API函数解析出hook变量包含的函数并执行。好理解吧,一句话就说清楚了hook机制,但是实现并不简单,最关键的就是那个内置API函数解析hook变量。后面笔者先不给大家掰Wordpress源码的hook解析过程了,那个太复杂,考虑的方面很多,学习理解起来比较困难。这里我们采用更简单直接的解说方式,利用原生PHP函数编写个简单的hook机制plugin插件管理类,和大家一起更加直观的理解钩子机制的原理过程。
下面代码的基本过程是:PluginManager内部有hook键值数组_listener()。
1、plugin注册到PluginManager类的包含hook键值的监听数组_listener()中,
2、PluginManager类实例对象调用trigger函数实现加载所有plugin插件并执行功能方法。
具体过程详见如下代码注释分析:
<?php // 已注册插件管理核心类
class PluginManager
{
/**
* 监听数组,保存所有已注册插件的类私有的核心数组变量,数组的键名是钩子名,值是对应的插件信息
* @var array private $_listeners
*/
private $_listeners = array();
/**
* 默认构造函数的作用是通过get_active_plugins()读取plugins目录下所有已激活插件信息
* 同时初始化这些插件,注册到核心类PluginManager数组变量$_listeners中
*
* @var array $plugins
* @return void
*/
public function __construct()
{
//这里$plugins数组包含我们获取已激活的所有插件信息,通过get_active_plugins()函数获取具体信息
$plugins = array();
$plugins = $this->get_active_plugins(); if(is_array($plugins) && !empty($plugins) && count($plugins) > 0)
{
foreach($plugins as $plugin)
{
// 约定每个插件类的名字为如下格式,例如DemoActions;
$class = $plugin['name'].'Actions';
if(class_exists($class))
{
//初始实例化已激活插件,$this代表PluginManager实例为参数
new $class($this);
}
}
}
} /**
* 注册需要监听插件的功能方法绑定到hook钩子,并把hook钩子加入到$_listeners数组
*
* @param string $hook 钩子变量,就是数组的键名,每个钩子可以绑定多个plugin插件类
* @param object $plugin 插件变量,get_class($plugin)获取插件对应的类
* @param string $method 插件$plugin类对应的功能方法
*/
function register($hook, $plugin, $method)
{
//获取插件实现的功能方法
$key = get_class($plugin).'->'.$method;
//echo $key.'<br>'; //这里可以测试$key的值是否是实例方法引用;
//将插件的实例对象和功能方法保存入对应键值为hook名的监听数组中
$this->_listeners[$hook][$key] = array($plugin, $method);
} /**
* 返回已激活的所有插件名称和路径,读取plugins目录下所有已激活插件信息
*
* @return array() $plugins 返回数组包含每组插件$name:插件名称,也是php文件名;$directory:插件所在路径
*/
function get_active_plugins()
{
$dir = dirname(__FILE__).DIRECTORY_SEPARATOR.'plugins';
$filesnames = scandir($dir);
$plugins = array();
foreach($filesnames as $filename)
{
if($filename!='.' &&$filename!='..')
{
$plugins[] = array(
'name' => strstr($filename,'.', true),
'directory'=>$dir);
}
}
return $plugins;
} /**
* 触发一个钩子名称下所有的插件自定义功能方法
*
* @param string $hook 钩子的名称
* @param mixed $data 输入钩子内对应插件自定义方法的参数,默认为空
* @return mixed
*/
function trigger($hook, $data='')
{
//查看要实现的钩子,是否在监听数组之中
if (isset($this->_listeners[$hook]) && is_array($this->_listeners[$hook]) && count($this->_listeners[$hook]) > 0)
{
// 循环调用hook钩子所有插件功能方法
foreach ($this->_listeners[$hook] as $listener)
{
// 取出插件实例对象类名
$class = $listener[0];
// 取出插件实例对象自定义的功能方法
$method = $listener[1];
if(method_exists($class,$method))
{
// 动态调用hook钩子下所有插件的功能方法,这里$data为可无的方法参数
$class->$method($data);
}
}
}
} }
下面的是插件类DemoActions,其解析函数内包含对pluginManager对象的引用,对应的插件文件是Demo.php,该插件自定义功能方法为sayHello()。
// 插件类,约定必须包含固定格式解析函数
class DemoActions
{
/**
* 解析函数的参数是pluginManager类的引用实例
* 函数调用pluginManager实例的register方法注册这个插件
*/
function __construct(&$pluginManager)
{
/* hookdemo参数是钩子的名称
* $this是Demo_actions类的实例
* say_hello参数是此插件的功能方法
*/
$pluginManager->register('hookdemo', $this, 'sayHello');
} // 这里是自定义的插件功能方法
function sayHello()
{
echo '<br>Hello World<br>';
}
}
实际使用的时候,编辑如下代码程序:
//实际应用程序
$pluginManager = new PluginManager; //插件管理类实例化对象
$pluginManager->trigger('hookdemo',''); //启动绑定到hookdemo钩子的所有插件功能;
至此,我们就完整的实现了hook钩子绑定插件信息及如何利用hook钩子执行插件自定义功能方法的原理。Wordpress的hook钩子原理与此类似,理解了上面的代码,再逐步深入理解Wordpress源码的钩子机制就会更加如鱼得水。
Wordpress解析系列之PHP编写hook钩子原理简单实例的更多相关文章
- WordPress 插件机制的简单用法和原理(Hook 钩子)
WordPress 的插件机制实际上只的就是这个 Hook 了,它中文被翻译成钩子,允许你参与 WordPress 核心的运行,是一个非常棒的东西,下面我们来详细了解一下它. PS:本文只是简单的总结 ...
- 【原】Android热更新开源项目Tinker源码解析系列之三:so热更新
本系列将从以下三个方面对Tinker进行源码解析: Android热更新开源项目Tinker源码解析系列之一:Dex热更新 Android热更新开源项目Tinker源码解析系列之二:资源文件热更新 A ...
- ETL利器Kettle实战应用解析系列一【Kettle使用介绍】
本系列文章主要索引如下: 一.ETL利器Kettle实战应用解析系列一[Kettle使用介绍] 二.ETL利器Kettle实战应用解析系列二 [应用场景和实战DEMO下载] 三.ETL利器Kettle ...
- C# Hook钩子实例代码之截取键盘输入,需要的朋友可以参考下
一.关于本文 以最通俗的语言说明钩子的使用方法,具体到钩子的详细介绍可以参照下面的网址: http://www.microsoft.com/china/community/program/origin ...
- {Django基础十之Form和ModelForm组件}一 Form介绍 二 Form常用字段和插件 三 From所有内置字段 四 字段校验 五 Hook钩子方法 六 进阶补充 七 ModelForm
Django基础十之Form和ModelForm组件 本节目录 一 Form介绍 二 Form常用字段和插件 三 From所有内置字段 四 字段校验 五 Hook钩子方法 六 进阶补充 七 Model ...
- Android源码解析系列
转载请标明出处:一片枫叶的专栏 知乎上看了一篇非常不错的博文:有没有必要阅读Android源码 看完之后痛定思过,平时所学往往是知其然然不知其所以然,所以为了更好的深入Android体系,决定学习an ...
- 【转】Hook钩子C#实例
[转]Hook钩子C#实例 转过来的文章,出处已经不知道了,但只这篇步骤比较清晰,就贴出来了. 一.写在最前 本文的内容只想以最通俗的语言说明钩子的使用方法,具体到钩子的详细介绍可以参照下面的网址: ...
- HOOK钩子教程
[转载]HOOK钩子教程 http://blog.sina.com.cn/s/blog_675049f701019ka9.html(原贴) 先留着,好好学一学! 原文地址:HOOK钩子教程作者:X_T ...
- Kotlin协程解析系列(上):协程调度与挂起
vivo 互联网客户端团队- Ruan Wen 本文是Kotlin协程解析系列文章的开篇,主要介绍Kotlin协程的创建.协程调度与协程挂起相关的内容 一.协程引入 Kotlin 中引入 Corout ...
随机推荐
- Jmeter-BeanShell PostProcessor提取请求及响应结果并保存到本地文件
1.新建一个本地csv文件,存放请求需要使用的变量值account,password,并配置CSV Data Set Config 2.添加一个HTTP请求 3.添加正则提取器用来提取响应结果中的re ...
- linux下安装QT5:error: unrecognized command line option ‘-fuse-ld=gold’
安装qt时在执行./configure时报错:error: unrecognized command line option '-fuse-ld=gold' 这个错误是qt的一个bug. 在装有gol ...
- 【LeetCode】309. Best Time to Buy and Sell Stock with Cooldown
题目: Say you have an array for which the ith element is the price of a given stock on day i. Design a ...
- jQuery(二) jQuery对Ajax的使用
学习使我快乐!嘿 --WH 一.jQuery使用Ajax 想要了解jQuery如何使用Ajax,并且体会到它所带来的方便性,那么就得了解原始的Ajax是如何编写的,是怎样的繁琐,然后和Jquery的代 ...
- sizzle选择器的使用
<!doctype html> <html> <head> <meta charset="utf-8"> <title> ...
- angular4.0 父子组建之间的相互通信
父组建---->子组建 传递信息 首先先通过angular脚手架生成两个基本组件,有一个好处是 会自动关联到跟模版,节约时间,而且还是偷懒 ng generate component compo ...
- 实现一个简单的Log框架
实际上算不上框架,只是自己对日志框架的一点理解. 核心接口:Logger,供调用者完成不同等级的日志输出 package com.lichmama.log.service; public interf ...
- MySQL缓存之Qcache与buffer pool对比
Q:innodb buffer pool和Qcache的缓存区别? A: 1.Qcacche缓存的是SQL语句及对应的结果集,缓存在内存,最简单的情况是SQL一直不重复,那Qcache的命令率肯定是0 ...
- Jenkins-FQA
1.svn url不能设置成文件路径,而应该是"文件夹"路径,也就是说必须是目录. 示例: Checking out https://svn.gw.com.cn:10000/svn ...
- jsp获取当前日期,包括星期几
<%@ page language="java" pageEncoding="GB2312" %> <html> <head> ...