1、get、set
这两个方法是为在类和他们的父类中没有声明的属性而设计的
- __get( $property ) 当调用一个未定义的属性时访问此方法
- __set( $property, $value ) 给一个未定义的属性赋值时调用
这里的没有声明包括访问控制为proteced,private的属性(即没有权限访问的属性)
2、isset、unset
- __isset( $property ) 当在一个未定义的属性上调用isset()函数时调用此方法
- __unset( $property ) 当在一个未定义的属性上调用unset()函数时调用此方法
与get方法和set方法相同,这里的没有声明包括访问控制为proteced,private的属性(即没有权限访问的属性)
3、__call
call( $method, $arg_array ) 当调用一个未定义(包括没有权限访问)的方法是调用此方法
**4、autoload
autoload 函数,使用尚未被定义的类时自动调用。通过此函数,脚本引擎在 PHP 出错失败前有了最后一个机会加载所需的类。
注意: 在 autoload 函数中抛出的异常不能被 catch 语句块捕获并导致致命错误。
5、construct、destruct**
- __construct 构造方法,当一个对象被创建时调用此方法,好处是可以使构造方法有一个独一无二的名称,无论它所在的类的名称是什么,这样你在改变类的名称时,就不需要改变构造方法的名称
- __destruct 析构方法,PHP将在对象被销毁前(即从内存中清除前)调用这个方法
默认情况下,PHP仅仅释放对象属性所占用的内存并销毁对象相关的资源.,析构函数允许你在使用一个对象之后执行任意代码来清除内存,当PHP决定你的脚本不再与对象相关时,析构函数将被调用.,在一个函数的命名空间内,这会发生在函数return的时候,对于全局变量,这发生于脚本结束的时候,如果你想明确地销毁一个对象,你可以给指向该对象的变量分配任何其它值,通常将变量赋值勤为NULL或者调用unset。
6、__clone
PHP5中的对象赋值是使用的引用赋值,使用clone方法复制一个对象时,对象会自动调用clone魔术方法,如果在对象复制需要执行某些初始化操作,可以在clone方法实现。
7、__toString
toString方法在将一个对象转化成字符串时自动调用,比如使用echo打印对象时,如果类没有实现此方法,则无法通过echo打印对象,否则会显示:Catchable fatal error: Object of class test could not be converted to string in,此方法必须返回一个字符串。
在PHP 5.2.0之前,toString方法只有结合使用echo() 或 print()时 才能生效。PHP 5.2.0之后,则可以在任何字符串环境生效(例如通过printf(),使用%s修饰符),但 不能用于非字符串环境(如使用%d修饰符)。从PHP 5.2.0,如果将一个未定义toString方法的对象 转换为字符串,会报出一个E_RECOVERABLE_ERROR错误。
**8、sleep、__wakeup**
- __sleep 串行化的时候用
- __wakeup 反串行化的时候调用
serialize() 检查类中是否有魔术名称 sleep 的函数。如果这样,该函数将在任何序列化之前运行。它可以清除对象并应该返回一个包含有该对象中应被序列化的所有变量名的数组。
使用 sleep 的目的是关闭对象可能具有的任何数据库连接,提交等待中的数据或进行类似的清除任务。此外,如果有非常大的对象而并不需要完全储存下来时此函数也很有用。
相反地,unserialize() 检查具有魔术名称 wakeup 的函数的存在。如果存在,此函数可以重建对象可能具有的任何资源。使用 wakeup 的目的是重建在序列化中可能丢失的任何数据库连接以及处理其它重新初始化的任务。
9、__set_state
当调用var_export()时,这个静态 方法会被调用(自PHP 5.1.0起有效)。本方法的唯一参数是一个数组,其中包含按array(’property’ => value, …)格式排列的类属性。
10、__invoke
当尝试以调用函数的方式调用一个对象时,invoke 方法会被自动调用。PHP5.3.0以上版本有效
**11、callStatic**
它的工作方式类似于 call() 魔术方法,callStatic() 是为了处理静态方法调用,PHP5.3.0以上版本有效,PHP 确实加强了对 callStatic() 方法的定义;它必须是公共的,并且必须被声明为静态的。同样,call() 魔术方法必须被定义为公共的,所有其他魔术方法都必须如此。
对象序列化是一种持久化对象的方式,并且序列化对象只会保留对象属性
$car = new Car();
$car->setBrand("领克01");
// 将对象序列化为字符串后保存到文件
$string = serialize($car);
file_put_contents("car", $string);
//读取解析
$content = file_get_contents("car");
$object = unserialize($content);
echo "汽车品牌:" . $object->getBrand() . PHP_EOL;
以上便是序列化和反序列化,相对的就是sleep、wakeup
call,callStatic**
当在指定对象上调用一个不存在的成员方法时,如果该对象包含 __call
魔术方法,则尝试调用该方法作为兜底,与之类似的,当在指定类上调用一个不存在的静态方法,如果该类包含 __callStatic
方法,则尝试调用该方法作为兜底。
__invoke()
__invoke
魔术方法会在以函数方式调用对象时执行,还是以 Car 为例,我们在其中定义 __invoke
魔术方法如下:
$car = new Car();
$car('宝马');
__clone()
当对象被克隆的时候被调用
直接克隆对象会有问题:
通过 clone
拷贝的对象普通属性不再相互污染,但是嵌套的对象属性依然存在这个互相影响的问题,因此,我们把引用赋值和 clone 拷贝统统称之为「浅拷贝」,只有嵌套的对象属性也不相互污染的拷贝才是真正相互对立的「深拷贝」。要实现这种深拷贝,就要用到我们前面提到的 __clone
魔术方法。
<?php
class Engine
{
public $num = 4;
}
class Car
{
public $brand;
public $power;
public $engine;
public function __clone()
{
$this->engine = clone $this->engine;
}
}
$benz = new Car();
$benz->brand = '奔驰';
$benz->power = '汽油';
$engine = new Engine();
$benz->engine = $engine;
$lnykco02 = clone $benz;
$lnykco02->brand = '领克02';
$lnykco02->power = '电池';
$lnykco02->engine->num = 3;
var_dump($benz);
var_dump($lnykco02);