首页 > 编程知识 正文

php钩子hook实现(hook钩子技术)

时间:2023-12-17 12:25:53 阅读:316469 作者:HTKK

本文目录一览:

PHP中如何实现Hook机制

给你推荐2个方法

类的__call 和__callstatic 。 这2个魔术方法是当调用类的内部方法不存在时执行。__call是实例化类之后调用方法不存在。__callstatic 是调用静态方法不存在,它里面的参数包含了方法名和传递的参数。这样就可以利用这2个方法去实现hook机制。根据方法名和参数去找指定位置的文件和方法执行。use_call_func和use_call_func_array 具体看文档

是反射机制。php已经很早就有反射机制了。利用类的反射机制。去实现hook。其实原理和1方法差不多。但是实现的方法比较高级而已。

php中的hook

Prestashop Quick Primer

Prestashop Quick Primer

方便给网页设计者了解 Prestashop的主要结构

看了前面一章的介绍文章,我想你也迫不及待想要看看主要实现的方法吧?别急,现在就开始。记得转载的时候注明:哦!

Prestashop,像所有的PHP脚本语言一样会加载本地服务器上的 index.php 文件。如果你的 Prestashop 商店程序是放在mystore.com,当客户端输入URL : 会自动加载到index这个页面。

为了了解咋们的prestashop shop是如何运行的,我们需要去了解下index文件是如何去工作的。

Index.php

prestashop index.php文件中包含着以下的代码 :

1. ?php

2. include(dirname(__FILE__).'/config/config.inc.php');

3. include(dirname(__FILE__).'/header.php');

4.

5. $smarty-assign('HOOK_HOME', Module::hookExec('home'));

6. $smarty-display(_PS_THEME_DIR_.'index.tpl');

7.

8. include(dirname(__FILE__).'/footer.php');

9.

10. ?

在 index.php 文件的开始包含着一个 config.inc.php 的文件,这个文件时主要处理我们的主要设置还有一些程序相关其它设置,config.inc.php 会检查是否所有的东西已经安装好了,并且确认了我们的默认模版和默认目录,还有路径、订单的状态还有其它的一些设置。或者现在你已经在抱怨着什么时候才能开始学习到相关的模版制作,现在我们只知道 index.php 中包含了什么而已。

然而,作为一个网页设计者,你可能需要及时地注意到了网页上的错误信息,因此现在就要用到刚才所提到的 config.inc.php 文件了,在这个文件最上面几行,你可以看到:

1. @ini_set('display_errors', 'off');

2. 你可以选择把它打开或关闭

3. @ini_set('display_errors', 'on');

现在你可以关闭错误信息报告了。不过要注意在下次正式安装前要吧它改回来。

下一步,来看看 header.php 的代码:

1. ?php

2.

3. // P3P Policies ()

4. header('P3P: CP="IDC DSP COR CURa ADMa OUR IND PHY ONL COM STA"');

5.

6. require_once(dirname(__FILE__).'/init.php');

7.

8. /* CSS */

9. $css_files[_THEME_CSS_DIR_.'global.css'] = 'all';

10.

11. /* Hooks are voluntary out the initialize array (need those variables already assigned) */

12. $smarty-assign(array(

13. 'HOOK_HEADER' = Module::hookExec('header'),

14. 'HOOK_LEFT_COLUMN' = Module::hookExec('leftColumn'),

15. 'HOOK_TOP' = Module::hookExec('top'),

16. 'static_token' = Tools::getToken(false),

17. 'token' = Tools::getToken(),

18. 'priceDisplayPrecision' = _PS_PRICE_DISPLAY_PRECISION_,

19. 'content_only' = intval(Tools::getValue('content_only'))

20. ));

21.

22. if(isset($css_files) AND !empty($css_files)) $smarty-assign('css_files', $css_files);

23. if(isset($js_files) AND !empty($js_files)) $smarty-assign('js_files', $js_files);

24.

25. /* Display a maintenance page if shop is closed */

26. if (isset($maintenance) AND (!isset($_SERVER['REMOTE_ADDR']) OR $_SERVER['REMOTE_ADDR'] != Configuration::get('PS_MAINTENANCE_IP')))

27. {

28. header('HTTP/1.1 503 temporarily overloaded');

29. $smarty-display(_PS_THEME_DIR_.'maintenance.tpl');

30. exit;

31. }

32.

33. $smarty-display(_PS_THEME_DIR_.'header.tpl');

34.

35. ?

接下来我们可以看到这句代码:

1. $smarty-assign('HOOK_HOME', Module::hookExec('home'));

这里将会指定分配hook(我们通常都会第一时间把hook给指定分配了,你也可以指定哪些模块在你网站中出现)

1. $smarty-display(_PS_THEME_DIR_.'index.tpl');

这一行代码是说明 index.tpl 相当于默认主题模版目录的位置(你可以定义或者添加你的主题,backoffice preferancesappearance)

最后我们再来看看footer.php

1. include(dirname(__FILE__).'/footer.php');

这一行代码include了我们的footer.php文件,看下footer.php文件代码:

1. ?php

2.

3. if (isset($smarty))

4. {

5. $smarty-assign(array(

6. 'HOOK_RIGHT_COLUMN' = Module::hookExec('rightColumn'),

7. 'HOOK_FOOTER' = Module::hookExec('footer'),

8. 'content_only' = intval(Tools::getValue('content_only'))));

9. $smarty-display(_PS_THEME_DIR_.'footer.tpl');

10. }

11.

12. ?

尽管我们现在了解到了程序的配置设置,hooks还有不同的php文件包含,但是你会发现,目前为止,还没出现我们所熟悉的HTML代码。其实这个就是我们所有templates(tpl)的来源入口。tpl文件里面包含着HTML还有smarty模版语言。这些tpl文件大多都在你的模版目录下,有些也在每个功能模块目录下。

现在来看看header.php文件,你会看到里面有这些代码:

1. $css_files[_THEME_CSS_DIR_.'global.css'] = 'all';

2. if(isset($css_files) AND !empty($css_files)) $smarty-assign('css_files', $css_files);

3. if(isset($js_files) AND !empty($js_files)) $smarty-assign('js_files', $js_files);

4.

5. $smarty-display(_PS_THEME_DIR_.'maintenance.tpl'); - Will display a maintenance page called maintenance.tpl if your shop is closed */

6.

7. $smarty-display(_PS_THEME_DIR_.'header.tpl');

我们的global.css和css、javascript文件都被包含在这里,html的显示来自于header.tpl文件。

以此类推,index.php 和footer.php也有这对应的index.tpl和footer.tpl(其它的文件也是这样的命名规则)

现在目前看起来好像有点混乱了,不用担心,坚持下去,接下来会变得简单些的。

好了,从这里开始打段,总结下:index.php文件包含着header.php,index.tpl和footer.tpl,header.php和footer.php的tpl文件也分别取自他们本身的文件名。

我们主要的HTML代码都来源于这三个tpl文件。

现在我们来打开header.tpl,index,tpl 还有footer.tpl,这里对于web设计师会相对熟悉点

Header.tpl

1. !DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN" ""

2. html xmlns="" xml:lang="{$lang_iso}"

3. head

4. base href="{$protocol}{$smarty.server.HTTP_HOST|escape:'htmlall':'UTF-8'}{$base_dir}" /

5. title{$meta_title|escape:'htmlall':'UTF-8'}/title

6. {if isset($meta_description) AND $meta_description}

7. meta name="description" content="{$meta_description|escape:htmlall:'UTF-8'}" /

8. {/if}

9. {if isset($meta_keywords) AND $meta_keywords}

10. meta name="keywords" content="{$meta_keywords|escape:htmlall:'UTF-8'}" /

11. {/if}

12. meta http-equiv="Content-Type" content="application/xhtml+xml; charset=utf-8" /

13. meta name="generator" content="PrestaShop" /

14. meta name="robots" content="{if isset($nobots)}no{/if}index,follow" /

15. link rel="icon" type="image/vnd.microsoft.icon" href="{$img_ps_dir}favicon.ico" /

16. link rel="shortcut icon" type="image/x-icon" href="{$img_ps_dir}favicon.ico" /

17. {if isset($css_files)}

18. {foreach from=$css_files key=css_uri item=media}

19. link href="{$css_uri}" rel="stylesheet" type="text/css" media="{$media}" /

20. {/foreach}

21. {/if}

22. script type="text/javascript" src="{$base_dir}js/tools.js"/script

23. script type="text/javascript"

24. var baseDir = '{$base_dir}';

25. var static_token = '{$static_token}';

26. var token = '{$token}';

27. var priceDisplayPrecision = {$priceDisplayPrecision*$currency-decimals};

28. /script

29. script type="text/javascript" src="{$base_dir}js/jquery/jquery-1.2.6.pack.js"/script

30. script type="text/javascript" src="{$base_dir}js/jquery/jquery.easing.1.3.js"/script

31. {if isset($js_files)}

32. {foreach from=$js_files item=js_uri}

33. script type="text/javascript" src="{$js_uri}"/script

34. {/foreach}

35. {/if}

36. {$HOOK_HEADER}

37. /head

38.

39. body {if $page_name}id="{$page_name|escape:'htmlall':'UTF-8'}"{/if}

40. {if !$content_only}

41. div id="page"

42.

43. !-- Header --

44. div

45. h1 id="logo"a href="{$base_dir}" title="{$shop_name|escape:'htmlall':'UTF-8'}"img src="{$img_ps_dir}logo.jpg" alt="{$shop_name|escape:'htmlall':'UTF-8'}" //a/h1

46. div id="header"

47. {$HOOK_TOP}

48. /div

49. /div

50.

51. !-- Left --

52. div id="left_column" class="column"

53. {$HOOK_LEFT_COLUMN}

54. /div

55.

56. !-- Center --

57. div id="center_column"

58. {/if}

Index.tpl

1. {$HOOK_HOME}

Footer.tpl

1. {if !$content_only}

2. /div

3.

4. !-- Right --

5. div id="right_column" class="column"

6. {$HOOK_RIGHT_COLUMN}

7. /div

8.

9. !-- Footer --

10. div id="footer"{$HOOK_FOOTER}/div

11. /div

12. {/if}

13. /body

14. /html

现在我们来看看homepage的源文件:

Index.tpl

1. !DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.1//EN" ""

2. html xmlns="" xml:lang="en"

3. head

4. base href="" /

5. titlePrestashop demo/title

6. meta name="description" content="Shop powered by PrestaShop" /

7. meta name="keywords" content="shop, prestashop" /

8. meta http-equiv="Content-Type" content="application/xhtml+xml; charset=utf-8" /

9. meta name="generator" content="PrestaShop" /

10. meta name="robots" content="index,follow" /

11. link rel="icon" type="image/vnd.microsoft.icon" href="/prestashop/img/favicon.ico" /

12. link rel="shortcut icon" type="image/x-icon" href="/prestashop/img/favicon.ico" /

13. link href="/prestashop/themes/commotion/css/global.css" rel="stylesheet" type="text/css" media="all" /

14. script type="text/javascript" src="/prestashop/js/tools.js"/script

15. script type="text/javascript"

16. var baseDir = '/prestashop/';

17. var static_token = '8d6a4d79d983512d770333775a7d8d24';

18. var token = '521aacf70fb52a35b4f94e01366c4c64';

19. var priceDisplayPrecision = 2;

20. /script

21. script type="text/javascript" src="/prestashop/js/jquery/jquery-1.2.6.pack.js"/script

22. script type="text/javascript" src="/prestashop/js/jquery/jquery.easing.1.3.js"/script

23. /head

24. body id="index"

25. div id="page"

26.

27. !-- Header --

28. div

29. h1 id="logo"a href="/prestashop/" title="Prestashop demo"img src="/prestashop/img/logo.jpg" alt="Prestashop demo" //a/h1

30. div id="header"

31.

32. /div

33. /div

34.

35. !-- Left --

36. div id="left_column" class="column"

37.

38. /div

39.

40. !-- Center --

41. div id="center_column"

42. /div

43.

44. !-- Right --

45. div id="right_column" class="column"

46.

47. /div

48.

49. !-- Footer --

50. div id="footer"

51. /div

52. /div

53. /body

54. /html

可能你会注意到了index.tpl 只有一行{$HOOK_HOME}。这里是因为index文件只在homepage上显示我们所指定的模块,我们的目录还有产品页将会显示其它的模块。

当然,你可能也会发现我们每个页面都有一个不同的body id,方便了我们用css去定义他们单独的每个页面,或许你以后还想在body上加上些class,那么这句是从哪句代码得来的呢?看看header.tpl,这段代码就是用来实现它的:

1. body {if $page_name} id="{$page_name|escape:'htmlall':'UTF-8'}"{/if} and it is this code that gets the page name. So category.tpl will have the body id="category"

php中关于hook用途的问题

PHP 写插件 其中的HOOK 只是别人想的一个词语,术语而已

并无实际意义

在WINDOWS编程中 HOOK==钩子

是一种消息机制的一种

意思就是:蚊子叮了你一口,你的大脑神经接收到了,于是指挥肢体,去赶走还是拍死这个蚊子

这个大脑神经就是HOOK,系统接收用户操作,而HOOK则监视某一满足条件

用户输入A

,系统接收到A,而HOOK定义接收B,那么输入A,则不做操作

当输入B了,则激活HOOK,进行执行制定代码

PHP的插件HOOK,比如你添加一个插件,把这个插件信息提前告诉HOOK,然后HOOK去监视用户

当用户调用这个插件,HOOK就感知到了

于是就把插件功能实现出来

thinkphp hook 怎么写

ThinkPHP3.2 扩展--钩子,HOOK

之前写到TP3.1的行为扩展是tag();在TP3.2中引入了另一种说法—:钩子。

我们来看一下TP3.2中的钩子这个东西:

一:文件流程:

1:/index.php -require './ThinkPHP/ThinkPHP.php';

2:/ThinkPHP/ThinkPHP.php—-require CORE_PATH.'Think'.EXT; ThinkThink::start();

3:/ThinkPHP/Library/Think/Think.class.php—–App::run();

4:/ThinkPHP/Library/Think/App.class.php 。到这里基本流程就走完了,(这里不说细节);

二:代码:

1:看一下 App::run()方法:

// 应用初始化标签

Hook::listen('app_init');

App::init();

// 应用开始标签

Hook::listen('app_begin');

// Session初始化

if(!IS_CLI){

session(C('SESSION_OPTIONS'));

}

// 记录应用初始化时间

G('initTime');

App::exec();

// 应用结束标签

Hook::listen('app_end');

return ;

其中的Hook::listen(”)就是用来执行钩子的,我们可以在app_init这个安插的位置用来获取应用中安装的插件。

看一下Hook::listen();

/**

* 监听标签的插件

* @param string $tag 标签名称

* @param mixed $params 传入参数

* @return void

*/

static public function listen($tag, $params=NULL) {

if(isset(self::$tags[$tag])) {

if(APP_DEBUG) {

G($tag.'Start');

trace('[ '.$tag.' ] --START--','','INFO');

}

foreach (self::$tags[$tag] as $name) {

APP_DEBUG G($name.'_start');

$result = self::exec($name, $tag,$params);

if(APP_DEBUG){

G($name.'_end');

trace('Run '.$name.' [ RunTime:'.G($name.'_start',$name.'_end',6).'s ]','','INFO');

}

if(false === $result) {

// 如果返回false 则中断插件执行

return ;

}

}

if(APP_DEBUG) { // 记录行为的执行日志

trace('[ '.$tag.' ] --END-- [ RunTime:'.G($tag.'Start',$tag.'End',6).'s ]','','INFO');

}

}

return;

}

其中关键是:self::exec($name, $tag,$params); 看一下exec的代码:

/**

* 执行某个插件

* @param string $name 插件名称

* @param string $tag 方法名(标签名)

* @param Mixed $params 传入的参数

* @return void

*/

static public function exec($name, $tag,$params=NULL) {

if(false === strpos($name,'\')) {

// 插件(多个入口)

$class = "Addons\{$name}\{$name}Addon";

}else{

// 行为扩展(只有一个run入口方法)

$class = $name.'Behavior';

$tag = 'run';

}

$addon = new $class();

return $addon-$tag($params);

}

最后还不是 new $class();进而return $addon-$tag($params); 又转到了具体钩子的代码方法。其实就是我们原本的调用class的方法,只不过经过别人的高度封装了。

三:那么问题来了,这个钩子有什么用呢? 怎么用?

这里以OneThink 的{:hook('AdminIndex')}为例,看一些别人是怎么用的。

在系统初始化到 Hook::listen('app_init'); 时,

把app_init的标签位扩展了,在tags.php的配置文件中有这么个东西:用于初始化插件(或者说是获取系统中安装的插件)

?php

return array(

'app_init'=array('CommonBehaviorInitHook')

);

一看就明白,无非就是读取持久化的信息,放到缓存或是其他的方式

// 行为扩展的执行入口必须是run

public function run($content){

if(isset($_GET['m']) $_GET['m'] === 'Install') return;

$data = S('hooks');

if(!$data){

$hooks = M('Hooks')-getField('name,addons');

foreach ($hooks as $key = $value) {

if($value){

$map['status'] = 1;

$names = explode(',',$value);

$map['name'] = array('IN',$names);

$data = M('Addons')-where($map)-getField('id,name');

if($data){

$addons = array_intersect($names, $data);

Hook::add($key,$addons);

}

}

}

S('hooks',Hook::get());

}else{

Hook::import($data,false);

}

}

当在程序执行到{:hook(‘AdminIndex’)}时—调用的是Hook::listen(‘AdminIndex’);

AdminIndex这个挂载点包含了三个插件:分别是:SiteStat, SystemInfo,DevTeam。

用一个循环来分别按顺序执行.

总结:钩子其实就是起到一个挂载点的作用,这个钩子挂在哪里,就可以在哪里执行,内容或功能就是挂载插件或类库的具体实现。这样实现的代码就有很大的灵活性,挂载点不变,挂的东西变量,功能也就相应的变化,是不是很灵活强大呀。

版权声明:该文观点仅代表作者本人。处理文章:请发送邮件至 三1五14八八95#扣扣.com 举报,一经查实,本站将立刻删除。