Php中钩子(Hook)的应用例子

我们先来回顾下原本的开发流程;产品汪搞出了一堆需求;当用户注册成功后需要发送短信、发送邮件等等;

然后聪明机智勇敢的程序猿们就一扑而上;把这些需求转换成代码扔在 用户注册成功 和 跳转到首页 之间;没有什么能够阻挡;充满创造力的猿们;

  1. class Test{
  2. public function index(){
  3. // 用户注册成功
  4. /*
  5. 此处是一堆发送短信的代码
  6. */
  7. /*
  8. 此处是一堆发送邮件的代码
  9. */
  10. /*
  11. 此处是一堆其他功能的代码
  12. */
  13. // 前往网站首页
  14. }
  15. }
  16. $test=new Test();
  17. $test->index();

如果每个功能都由不同的猿完成的话;首先面临的就是代码会很杂乱;配合起来会比较麻烦;那封装成函数吧;一方面会规范整洁写;另外方便重复调用;没有什么能够阻挡;充满创造力的猿们;

  1. class Test{
  2. public function index(){
  3. // 用户注册成功
  4. // 发送短信
  5. sendSms($phone);
  6. // 发送邮件
  7. sendSms($email);
  8. // 其他操作...
  9. // 前往网站首页
  10. }
  11. }
  12. /**
  13. * 发送短信通知
  14. * @param integer $phone 手机号
  15. */
  16. function sendSMS($phone){
  17. // 此处是发送短信的代码
  18. }
  19. /**
  20. * 发送邮件通知
  21. * @param string $email 邮箱地址
  22. */
  23. function sendEmail($email){
  24. // 此处是发送邮件的代码
  25. }

这时候运营喵表示;如果能在后台点点按钮就能设置是发邮件还是发短信;那想必是极好的;没有什么能够阻挡;充满创造力的猿们;

  1. class Test{
  2. public function index(){
  3. // 用户注册成功
  4. if ('如果设置了发送短信') {
  5. // 发送短信
  6. sendSms($phone);
  7. }
  8. if ('如果设置了发送邮件') {
  9. // 发送邮件
  10. sendSms($email);
  11. }
  12. // 其他操作...
  13. // 前往网站首页
  14. }
  15. }
  16. /**
  17. * 发送短信通知
  18. * @param integer $phone 手机号
  19. */
  20. function sendSMS($phone){
  21. // 此处是发送短信的代码
  22. }
  23. /**
  24. * 发送邮件通知
  25. * @param string $email 邮箱地址
  26. */
  27. function sendEmail($email){
  28. // 此处是发送邮件的代码
  29. }

在一个封闭企业环境下这样搞是没有问题的;然鹅;我们还有一位开放无私的猿领导要把程序开源出去造福其他猿类;

希望有更多的猿类来参与这个项目;共同开发功能;

如果大家都去改动这套程序;把自己的代码扔在 用户注册成功 和 跳转到首页之间;这显然是不靠谱的;想想都混乱的一塌糊涂;

那可不可以大家把自己写的代码放到某个目录下;然后系统自动的根据配置项把这些代码加载到 用户注册成功 和 跳转到首页 之间呢?

好先定义如下目录:

  1. ├─plugin // 插件目录
  2. │ ├─plugin1 // 插件1
  3. │ │ ├─config.php // 插件1的配置项
  4. │ │ ├─index.php // 插件1的程序处理内容
  5. │ ├─plugin2
  6. │ │ ├─config.php
  7. │ │ ├─index.php
  8. │ ├─plugin3
  9. │ │ ├─config.php
  10. │ │ ├─index.php
  11. │ ├─...
  12. ├─index.php // 业务逻辑

业务逻辑的代码:

  1. class Test{
  2. public function index(){
  3. // 用户注册成功
  4. // 获取全部插件
  5. $pluginList=scandir('./plugin/');
  6. // 循环插件 // 排除. ..
  7. foreach ($pluginList as $k => $v) {
  8. if ($v=='.' || $v=='..') {
  9. unset($pluginList[$k]);
  10. }
  11. }
  12. echo "简易后台管理<hr>";
  13. // 插件管理
  14. foreach ($pluginList as $k => $v) {
  15. // 获取配置项
  16. $config=include './plugin/'.$v.'/config.php';
  17. $word=$config['status']==1 ? '点击关闭' : '点击开启';
  18. echo $config['title'].'<a href="./index.php?change='.$v.'">'.$word.'</a><br />';
  19. }
  20. echo '<hr>';
  21. // 输出插件内容
  22. foreach ($pluginList as $k => $v) {
  23. // 获取配置项
  24. $config=include './plugin/'.$v.'/config.php';
  25. if ($config['status']==1) {
  26. include './plugin/'.$v.'/index.php';
  27. // 运行插件
  28. Hook::run($v);
  29. }
  30. }
  31. // 前往网站首页
  32. }
  33. }
  34. // 插件类
  35. class Hook{
  36. // 注册添加插件
  37. public static function add($name,$func){
  38. $GLOBALS['hookList'][$name][]=$func;
  39. }
  40. // 执行插件
  41. public static function run($name,$params=null){
  42. foreach ($GLOBALS['hookList'][$name] as $k => $v) {
  43. call_user_func($v,$params);
  44. }
  45. }
  46. }
  47. // 更改插件状态
  48. if (isset($_GET['change'])) {
  49. // 获取到配置项
  50. $config=include './plugin/plugin'.substr($_GET['change'],-1).'/config.php';
  51. // 如果是开启 那就关闭 如果是关闭 则开启
  52. $config['status']=$config['status']==1 ? 0: 1;
  53. // 将更改后的配置项写入到文件中
  54. $str="<?php \r\n return ".var_export($config,true).';';
  55. file_put_contents('./plugin/'.$_GET['change'].'/config.php', $str);
  56. header('Location:./');
  57. }
  58. $test=new Test();
  59. $test->index();

插件配置项代码:

  1. return array (
  2. 'status' => 1, // 定义状态 1表示开启 0表示关闭
  3. 'title' => '发送短信', // 插件的名称
  4. ;

插件的内容:

  1. Hook::add('plugin1',function(){
  2. echo '发送短信的内容<br />';
  3. });