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

 

转载于:https://www.cnblogs.com/yimuren/p/4566332.html

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值