php拦截器(魔术方法)
什么是PHP拦截器?
英文名称 “interceptor”,作用是 拦截 发送未定义的方法和属性的消息。
先看一段代码,定义了一个School类,实例化一个对象$obj,获取一个未定义的属性teacher,会发生什么呢?
<?php class School { } $obj = new School; var_dump($obj->teacher);
如果使用php命令行执行,可以看出,php报了一个未定义属性的Notice, $obj->teacher值是空值,所以打印出NULL
[root@localhost php]# php538 interceptor.php PHP Notice: Undefined property: School::$teacher in /usr/local/sina_mobile/apache/htdocs/php/interceptor.php on line 11 NULL
这里说明一下,php命令行执行和浏览器页面执行不一定完全一样,我们可以看下面截图,web server是 apache执行同样脚本时候,apache回告浏览器的是未带php报出的Notice,返回给浏览器的是一个200 OK,这里面存在这php和web server的交互以及web server和浏览器的交互问题。
针对前面的代码,我们想,如果访问teacher的时候,teacher属性不存在,我们可以拦截到对teacher的调用请求,并且做一些操作,那么这个就是php的拦截器,php拦截器提供了多种方法,这里__get方法就是访问未定义的属性时候被调用的。我们看下面代码和结果
<?php class School { function __get($var) { echo $var; echo "\r\n"; return "hello"; } public $name = "Tom"; } $obj = new School; var_dump($obj->teacher); var_dump($obj->name);
执行以下结果如下,说明一下,__get方法必须拥有一个参数,该参数就是调用的未定义的属性名称,函数会将return 结果赋值给该调用的未知属性。
[root@localhost php]# php538 interceptor.php teacher string(5) "hello" string(3) "Tom"
php提供了哪些拦截器方法
在有些文章中,把php拦截器一起称未php魔术方法,这里我们把这些方法做个整理
序号 | 方法名 | 作用 | 实现版本 |
1 | __get($property) | 访问未定义的属性时候,调用该方法 | >= PHP 5.3.0 |
2 | __set($property, $value) | 给未定义属性设置值的时候,调用该方法 | >= PHP 5.3.0 |
3 | __isset($property) | 对未定义属性调用isset()时,调用该方法 | >= PHP 5.1.0 |
4 | __unset($property) | 对未定义属性调用unset()时,调用该方法 | >= PHP 5.1.0 |
5 | __call($property, $args_array) | 调用未定义方法时,执行该方法 | >= PHP 5.3.0 |
6 | __autoload($classname) | 请求类时,先执行该方法 | >= PHP 5.3.0 |
7 | __construct($args) | 一个类中,只有一个构造函数,new 类的时候,构造函数优先被执行 | |
8 | __destruct() | 一个类中,只有一个析构函数,释放类的时候,析构函数被执行 | |
9 | __clone() | 使用clone创建对象副本时候,该方法被调用 | >= PHP 5.3.0 |
10 | __tostring() | __toString() 方法用于一个类被当成字符串时应怎样回应 | >= PHP 5.2.0 |
11 | __sleep() | serialize() 函数会检查类中是否存在一个魔术方法 __sleep()。如果存在,该方法会先被调用,然后才执行序列化操作 | >= PHP 5.3.0 |
12 | __wakeup() | unserialize() 会检查是否存在一个 __wakeup() 方法 | >= PHP 5.3.0 |
13 | __set_state | 调用 var_export() 导出类时,此静态 方法会被调用。 | >= PHP 5.3.0 |
14 | __invoke | 当尝试以调用函数的方式调用一个对象时,__invoke() 方法会被自动调用。 | >= PHP 5.3.0 |
15 | __callstatic() | 用静态方式中调用一个不可访问方法时,__callStatic() 会被调用 | |
16 | __debugInfo() | 使用var_dump()打印对象的时候,该方法被调用 | >= PHP 5.6.0 |
详细情况见 http://php.net/manual/zh/language.oop5.magic.php介绍
结论
在这里,可以把上表中序号1-5的方法称之为拦截器方法,后面称之为魔术方法,魔术方法可以理解成是提供给php程序员对php标准函数的"重写", 可以理解成zend虚拟机,在解析php脚本的时候,发现一些未定义的方法属性,或者发现一些提供用户重载的标准函数,如clone,serialize等时候,优先查看是否有需要自行的魔术方法。
参考文献:
《深入PHP面向对象、模式与实践》(第3版)/(美)Zandstra,M. 著,陈浩等译. 北京:人民邮电出版社,2011.7 ISDN 978-7-115-25624-9