php中的常用魔术方法总结

by admin on 2020年1月3日

常用的魔术方法有:__Tostring ()  __Call()  __autoLoad()  __
clone()  __GET()   __SET()    __isset()  __unset()
 
1.__Tostring()   用于定义输出对象引用时调用 
常用于打印一些对象的信息 必须有返回值
eg:有一个persion类
Persion per =new persion()
Echo per;    //直接调用会出错
我们可以在类的定义中添加__tostring()方法
复制代码 代码如下:
Function  __Tostring()
{
$str=this->$name.this->age;
Return $str;
}

只能说常用,因为魔术方法太多了,但是我们经常打交道的有限。

1、__get、__set

(1)_clone克隆
class Weather{
    public $isRain = "要下雨了<br>";
    public function howIsWeather(){
        echo "明天要下雨哦<br>";
    }
}

$nextDay = new Weather;

$otherDay = $nextDay;

echo $otherDay->isRain;  //使用对象调用属性,也不加$,与$this->调用相同
$otherDay->howIsWeather();

实例化对象的赋值是传递的对象的地址索引,两个变量指向同一个对象,属于浅拷贝,而clone与其含义一样,是创建一个独立的相同的对象,与之前的模板对象没有关系

class Weather{
    public $isRain = "要下雨了<br>";
    public function howIsWeather(){
        echo "明天要下雨哦<br>";
    }
}

$nextDay = new Weather;

$otherDay = clone $nextDay;

echo $otherDay->isRain;
$otherDay->howIsWeather();
$otherDay->isRain = "不会下雨了";
echo $nextDay->isRain;   //并不会被修改

retrieve 取回

2.__clone()对象的复制 引用赋值
$per1=$per2; 而这在内存中只有一块地址
而$per1=clone $per2   这时有两块内存地址

__construct():
类的构造方法,实例化类时调用:
class Bar{
    function __construct($name){
        $this->name = $name;
        echo "call construct";
    }
}
new Bar();//call construct;

__destruct()
类的析构方法,销毁对象时调用:
class Bar{
    function __destruct(){
        echo "destruct";
    }
}
$bar = new Bar();
$bar = null;//destruct
unset($bar);//destruct

__get()
__set()
__get()和__set()是一对,起到的作用仿佛是为每个成员变量都隐含的设置了get和set方法,
所以,在外部设置局部变量也成为可能,就像是访问public权限的变量一样。
class Bar{
    private $name;
    function __get($key){
        return $this->$key;
    }
    function __set($key,$value){
        $this->key = $value;
    }
}
$bar = new Bar();
$bar->name = 'foo';
echo $bar->name;//foo

__clone()
使用clone自动调用__clone()方法:
class Bar{
    function __clone(){
        echo "clone object";
    }
}
$bar = new Bar();
$foo = clone $bar;//clone object

这两个方法是为在类和他们的父类中没有声明的属性而设计的

(2)__call()

实例调用不存在的方法时会调用__call()魔术方法,并将不存在函数的方法名与参数传入call

3.__call()方法 当调用类实例中不存在的函数时自动执行 如果试图调用类中不存在的函数,会出现语法错误,为了能够友好的提示
我们可以在类中声明Call()方法;
复制代码 代码如下:
Function __call($funName,$argu)
{
Echo “名为”.$funName.”参数为”.printf($argh).”的函数不存在”,
}

关于__clone,官方有更好的说明

__get( $property ) 当调用一个未定义的属性时访问此方法

4.__autoLoad 自动加载使用的类文件  该函数是在引用的页面添加 我们都使用过这样情况,在页面中需要调用其他php文件,我们需要使用include方法
但是如果有几十个页面需要引用,未免太过繁琐,我们可以在该页面中使用autoload方法
复制代码 代码如下:
Function __autoload($className)
{
Include $className.”.php”;
}

__sleep()
序列化对象时调用

__wakeup()
反序列化对象时调用

__set( $property, $value ) 给一个未定义的属性赋值时调用

这样凡是引用到其他类的地方,都会自动引用该类文件 
前提类文件的名称必须是   类名.php

与一般的对象(如数组),类对象(自定义的class)的实例化和反实例化,尤其是单例(singleton)的序列化比较复杂,我们将用另一篇文章来说明。

这里的没有声明包括当使用对象调用时,访问控制为proteced,private的属性(即没有权限访问的属性)

5.__GET()   访问类中私有属性 如果类中的属性设置为私有属性,在类的实例中是无法访问的,但怎样才能访问呢?
我们就可以使用__GET()
Eg : 类中有
复制代码 代码如下:
Class person
{
Private $name;
Private $age;
}

__toString()
打印一个对象时调用:
class Bar{
    private $item = array(1,2,3);
    function toString(){
        echo implode($this->item,',');
    }
}

__invoke()
以函数的方式调用一个方法时自动调用__invoke():
class Bar{
    function __invoke($str)
    {
        // TODO: Implement __invoke() method.
        echo $str;
    }
}

$bar = new Bar;
$bar('foo');//foo

2、__isset、__unset

实例化 person per=new person()
Per->$name; //这样是取不到值的
但是如果我们在类中增加__GET方法
复制代码 代码如下:
Function __GET($proName)
{
Return this->$proName;
}

__isset( $property ) 当在一个未定义的属性上调用isset()函数时调用此方法

我们再次调用Per->$name 就可以访问了
这样做有人会提出疑问了,这样可以直接访问私有变量,和声明成公有的有什么区别呢?
如果声明成公有的,我们可以任意读取,如果是私有,如果我们增加了get方法,那么每次调用私有属性都会调用GET方法的内容,这样我们就可以在get方法中增加一些逻辑处理。

__unset( $property ) 当在一个未定义的属性上调用unset()函数时调用此方法

6.__SET()设置类中的私有属性 原理同上,我们可以再类中添加__SET()函数,每当通过调用类实例给私有属性赋值,都会执行__SET函数
函数原型: 复制代码 代码如下:
Function __SET($proName,$value)
{
This->$proName=$value;
}

与__get方法和__set方法相同,这里的没有声明包括当使用对象调用时,访问控制为proteced,private的属性(即没有权限访问的属性)

既然是方法赋值,我们就可以做一些逻辑处理

3、__call

7.__isset() 判断类中私有属性或方法是否存在时自动调用 首先我们先介绍一下isset
方法,该方法用于判定属性和方法是否存在,但是我们无法通过类类实例判断类中的某个私有属性是否存在
如果我们使用isset(per->$name);//返回值是false,但是$name属性的确存在,怎么解决呢?
解决方法: 1.将$name定义为私有属性
2.在类定义中添加
复制代码 代码如下:
Function __isset($proName)
{
Return  isset(this->$proName);//再类内部是可以访问私有属性的
}

__call( $method, $arg_array ) 当调用一个未定义的方法是调用此访求

这样的话我们再次调用isset($name);返回值就为true了;

这里的未定义的方法包括没有权限访问的方法

8.__unset()清除类中私有变量时自动调用 与之结合的是unset()
unset方法可以删除属性,当我们需要删除类中属性的时候,如果是公有属性我们可以直接
删除,但是如果是私有我们只通过该方法就无法实现了
怎样实现呢我们可以使用__unset()方法实现该功能我们需要在类中添加
复制代码 代码如下:
Function __unset($proName)
{
Unset(this->$proName);
}

4、__autoload

现在我们再调用unset($name);就可以删除person类中的私有属性$name了

__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()
魔术方法必须被定义为公共的,所有其他魔术方法都必须如此。

发表评论

电子邮件地址不会被公开。 必填项已用*标注

网站地图xml地图