0x00 引言
今日去面试,被面试官突然问了一句,php的魔术方法。一脸懵逼,没明白是啥。后来在回来的路上才想起,人间说的是那些前面带了__
的那些方法。一直以来我都叫两杆方法,原来这个是魔术方法啊。
因为面试官提到Java
的反射机制,我突然想到,想起来,这是php的面向对象很重要的一环。让我先从Java和PHP的对比说起。
Java和PHP面向对象的机制对比
Java
是一个面向对象的语言。设计之初,Java
就是面向对象的,Java
中的Object类,所有新建的类,如无指定默认就是继承这个类。继承这样的方式,我们可以称之为正向机制。而反向机制就是所谓反射。而在Object类中,有一个getClass方法获得Class。而Class中也有一些静态方法,来获取对象实例。比如很多人熟悉的初始化驱动程序的那行代码:
Class.forName("com.mysql.jdbc.Driver");
Note:
PHP 5中也引入了一个叫ReflectionClass的类,来实现PHP的反射机制。这个之后再写文介绍。
再说回PHP
,PHP
是一个函数式语言,是后来引入面向对象机制地。所以PHP
的面向对象机制是中是没有Object和Class这两个类的。PHP
中,每个魔术方法的调用,都是有时机的,也可说是事件,意思就是当发生那种情况时,这个魔术方法就会被调用,明确这一点也是重要的。下文我将魔术方法,分类来说明在面向对象中的作用。
0x01 构造与析构
__construct()
是构造方法,与Java
中无异。当new
关键字,创建一个对象时调用。值得注意的是,不会隐式地调用分类的构造方法,必须使用parent::__construct()
显式地调用。当然,如果一个没有定义构造方法,会从父类继承。
void __construct ([ mixed $args [, $... ]] )
__destruct()
是析构方法,当使用exit()
方法终止脚本,或者对象被GC回收的时候调用。
void __destruct ( void )
注意:不要试图在析构方法中停止脚本,那样会抛出一个异常。
PHP的垃圾回收机制:
PHP在PHP 5中开始引入垃圾回收机制。5.3之前只是单纯的适用引用计数方式,5.3之后才引入和Java的GC一样,引入避免引用成环的算法。
0x02 序列化和反序列化
PHP
中序列化和反序列化对象,是通过一组方法serialize()
和unserialize()
来实现。
serialize()
会先检查类中是否存在魔术方法__sleep()
,如果存在的情况下,该方法会被首先调用。此功能用于清理对象,并返回一个包含对象中所有应该序列化的变量名称和数组。如果不返回,则会报错。
unserialize()
会检查是否存在__wakeup()
方法, 如果存在则会县调用它,用于预先准备对象所需要的资源。
访问不可访问的属性
在给不可访问属性赋值时,__set() 会被调用。
public void __set ( string $name , mixed $value )
读取不可访问属性的值时,__get() 会被调用。
public mixed __get ( string $name )
当对不可访问属性调用 isset() 或 empty() 时,__isset() 会被调用。
public bool __isset ( string $name )
当对不可访问属性调用 unset() 时,__unset() 会被调用。
public void __unset ( string $name )
Note:
不可访问的属性包括不存在的属性。
访问不可访问的方法
在对象中调用一个不可访问方法时,__call() 会被调用。
public mixed __call ( string $name , array $arguments )
在静态上下文中调用一个不可访问方法时,__callStatic() 会被调用。
public static mixed __callStatic ( string $name , array $arguments )
Note:
同样的,不可访问的方法包括不存在的方法。
对象的特殊操作的魔术方法
- 将对象输出。
__toString()
在需要将对象输出时调用,比如echo $obj;
的时候。 - 当惨死以调用函数的方式调用对象时,会调用
__invoke()
方法。即使用new ClassName;
不带括号的方式声明一个对象,使用ClassName();
调用的是__invoke()
。 - 使用
clone
关键字复制对象完成时,新建对象的__clone()
方法会被调用。
调试信息
- 调用
var_export()
导出类时,__set_state()
会被调用。 - 使用
var_dump()
输出对象,调用的是__debugInfo()
。
总结
善用魔术方法,可以让你对象更强大。