本文实例讲述了php中钩子(hook)的原理与简单应用。分享给大家供大家参考,具体如下:
我们先来回顾下原本的开发流程;
产品汪搞出了一堆需求;
当用户注册成功后需要发送短信、发送邮件等等;
然后聪明机智勇敢的程序猿们就一扑而上;
把这些需求转换成代码扔在 用户注册成功 和 跳转到首页 之间;
没有什么能够阻挡;充满创造力的猿们;
<?php class Test{ public function index(){ // 用户注册成功 /* 此处是一堆发送短信的代码 */ /* 此处是一堆发送邮件的代码 */ /* 此处是一堆其他功能的代码 */ // 前往网站首页 } } $test=new Test(); $test->index();
如果每个功能都由不同的猿完成的话;
首先面临的就是代码会很杂乱;配合起来会比较麻烦;
那封装成函数吧;一方面会规范整洁写;另外方便重复调用;
没有什么能够阻挡;充满创造力的猿们;
<?php class Test{ public function index(){ // 用户注册成功 // 发送短信 sendSms($phone); // 发送邮件 sendSms($email); // 其他操作... // 前往网站首页 } } /** * 发送短信通知 * @param integer $phone 手机号 */ function sendSMS($phone){ // 此处是发送短信的代码 } /** * 发送邮件通知 * @param string $email 邮箱地址 */ function sendEmail($email){ // 此处是发送邮件的代码 }
这时候运营喵表示;
如果能在后台点点按钮就能设置是发邮件还是发短信;那想必是极好的;
没有什么能够阻挡;充满创造力的猿们;
<?php class Test{ public function index(){ // 用户注册成功 if ('如果设置了发送短信') { // 发送短信 sendSms($phone); } if ('如果设置了发送邮件') { // 发送邮件 sendSms($email); } // 其他操作... // 前往网站首页 } } /** * 发送短信通知 * @param integer $phone 手机号 */ function sendSMS($phone){ // 此处是发送短信的代码 } /** * 发送邮件通知 * @param string $email 邮箱地址 */ function sendEmail($email){ // 此处是发送邮件的代码 }
在一个封闭企业环境下这样搞是没有问题的;
然鹅;我们还有一位开放无私的猿领导要把程序开源出去造福其他猿类;
希望有更多的猿类来参与这个项目;共同开发功能;
如果大家都去改动这套程序;把自己的代码扔在 用户注册成功 和 跳转到首页 之间;
这显然是不靠谱的;想想都混乱的一塌糊涂;
那可不可以大家把自己写的代码放到某个目录下;
然后系统自动的根据配置项把这些代码加载到 用户注册成功 和 跳转到首页 之间呢?
好先定义如下目录
├─plugin // 插件目录
│ ├─plugin1 // 插件1
│ │ ├─config.php // 插件1的配置项
│ │ ├─index.php // 插件1的程序处理内容
│ ├─plugin2
│ │ ├─config.php
│ │ ├─index.php
│ ├─plugin3
│ │ ├─config.php
│ │ ├─index.php
│ ├─...
├─index.php // 业务逻辑
业务逻辑的代码:
<?php class Test{ public function index(){ // 用户注册成功 // 获取全部插件 $pluginList=scandir('./plugin/'); // 循环插件 // 排除. .. foreach ($pluginList as $k => $v) { if ($v=='.' || $v=='..') { unset($pluginList[$k]); } } echo "简易后台管理<hr>"; // 插件管理 foreach ($pluginList as $k => $v) { // 获取配置项 $config=include './plugin/'.$v.'/config.php'; $word=$config['status']==1 ? '点击关闭' : '点击开启'; echo $config['title'].'<a href="./index.php?change='.$v.'" rel="external nofollow" >'.$word.'</a><br />'; } echo '<hr>'; // 输出插件内容 foreach ($pluginList as $k => $v) { // 获取配置项 $config=include './plugin/'.$v.'/config.php'; if ($config['status']==1) { include './plugin/'.$v.'/index.php'; // 运行插件 Hook::run($v); } } // 前往网站首页 } } // 插件类 class Hook{ // 注册添加插件 public static function add($name,$func){ $GLOBALS['hookList'][$name][]=$func; } // 执行插件 public static function run($name,$params=null){ foreach ($GLOBALS['hookList'][$name] as $k => $v) { call_user_func($v,$params); } } } // 更改插件状态 if (isset($_GET['change'])) { // 获取到配置项 $config=include './plugin/plugin'.substr($_GET['change'],-1).'/config.php'; // 如果是开启 那就关闭 如果是关闭 则开启 $config['status']=$config['status']==1 ? 0: 1; // 将更改后的配置项写入到文件中 $str="<?php \\r\\n return ".var_export($config,true).';'; file_put_contents('./plugin/'.$_GET['change'].'/config.php', $str); header('Location:./'); } $test=new Test(); $test->index();
插件配置项代码:
<?php return array ( 'status' => 1, // 定义状态 1表示开启 0表示关闭 'title' => '发送短信', // 插件的名称 );
插件的内容:
没错;这就是插件的思想;
当然这只是一个超级简单的示例;
完整的插件机制要包括插件的类型、数据库、审核等等;
如果使用过wordpress或者国内的discuz;
你就会发现一个好的程序并不仅仅是自身多么优秀;
而且重要的就是设计的扩展性有多好;能多方便的让大家去扩展它的功能;
想对插件深入研究的话;建议去阅读wordpress、discuz的源代码;
更多关于PHP相关内容感兴趣的读者可查看本站专题:《php面向对象程序设计入门教程》、《PHP数组(Array)操作技巧大全》、《PHP基本语法入门教程》、《PHP运算与运算符用法总结》、《php字符串(string)用法总结》、《php+mysql数据库操作入门教程》及《php常见数据库操作技巧汇总》
希望本文所述对大家PHP程序设计有所帮助。
声明:本文内容来源于网络,版权归原作者所有,内容由互联网用户自发贡献自行上传,本网站不拥有所有权,未作人工编辑处理,也不承担相关法律责任。如果您发现有涉嫌版权的内容,欢迎发送邮件至:notice#nhooo.com(发邮件时,请将#更换为@)进行举报,并提供相关证据,一经查实,本站将立刻删除涉嫌侵权内容。