欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

PHP中的常見(jiàn)魔術(shù)方法功能作用及用法實(shí)例

 更新時(shí)間:2015年07月01日 16:38:31   投稿:junjie  
這篇文章主要介紹了PHP中的常見(jiàn)魔術(shù)方法功能作用及用法實(shí)例,本文講解了構(gòu)造函數(shù)和析構(gòu)函數(shù)__construct()和__desctruct()以及屬性重載(Property Overloading)__get()和、__set()、__isset()等等魔術(shù)方法,需要的朋友可以參考下

概述

在面向?qū)ο缶幊讨?,PHP提供了一系列的魔術(shù)方法,這些魔術(shù)方法為編程提供了很多便利。PHP中的魔術(shù)方法通常以__(兩個(gè)下劃線)開(kāi)始,并且不需要顯示的調(diào)用而是由某種特定的條件出發(fā)。這篇文章簡(jiǎn)單總結(jié)了PHP中提供的魔術(shù)方法。

開(kāi)始之前

在總結(jié)PHP的魔術(shù)方法之前先來(lái)定義兩個(gè)類,以便后邊示例使用:

復(fù)制代碼 代碼如下:

<?php
class Device {
    public $name;          
    public $battery;       
    public $data = array();
    public $connection;    
 
    protected function connect() {
        $this->connection = 'resource';
        echo $this->name . ' connected' . PHP_EOL;
    }
 
    protected function disconnect() {
        $this->connection = null;
        echo $this->name . ' disconnected' . PHP_EOL;
    }
}
 
class Battery {
    private $charge = 0;
 
    public function setCharge($charge) {
        $charge = (int)$charge;
        if($charge < 0) {
            $charge = 0;
        }
        elseif($charge > 100) {
            $charge = 100;
        }
        $this->charge = $charge;
    }
}
?>

Device類有四個(gè)成員屬性和兩個(gè)成員方法。Battery類有一個(gè)成員屬性和一個(gè)成員方法。

構(gòu)造函數(shù)和析構(gòu)函數(shù)

構(gòu)造函數(shù)和析構(gòu)函數(shù)分別在對(duì)象創(chuàng)建和銷毀時(shí)被調(diào)用。對(duì)象被“銷毀”是指不存在任何對(duì)該對(duì)象的引用,比如引用該對(duì)象的變量被刪除(unset)、重新賦值或腳本執(zhí)行結(jié)束,都會(huì)調(diào)用析構(gòu)函數(shù)。

__construct()

__construct()構(gòu)造函數(shù)是目前為止最經(jīng)常使用的函數(shù)。在創(chuàng)建對(duì)象時(shí),可以在構(gòu)造函數(shù)中做一些初始化工作??梢詾闃?gòu)造函數(shù)定義任意多個(gè)參數(shù),只要在實(shí)例化時(shí)傳入對(duì)應(yīng)個(gè)數(shù)的參數(shù)即可。構(gòu)造函數(shù)中出現(xiàn)的任何異常都會(huì)阻止對(duì)象的創(chuàng)建。

復(fù)制代碼 代碼如下:

class Device {
   public function  __construct(Battery $battery, $name) {
       $this->battery = $battery;
       $this->name = $name;
       $this->connect();
    }
}

上面的示例代碼中,Device類的構(gòu)造函數(shù)為成員屬性賦值并且調(diào)用了connect()方法。

復(fù)制代碼 代碼如下:

將構(gòu)造函數(shù)聲明為私有方法,可以防止在類外部創(chuàng)建對(duì)象,這在單利模式中經(jīng)常使用。

__desctruct()

析構(gòu)函數(shù)通常在對(duì)象被銷毀時(shí)調(diào)用,析構(gòu)函數(shù)不接收任何參數(shù)。經(jīng)常在析構(gòu)函數(shù)中執(zhí)行一些清理工作,比如關(guān)閉數(shù)據(jù)庫(kù)連接等。

屬性重載(Property Overloading)

有一點(diǎn)需要注意的是:PHP中的”重載”與其他大多數(shù)語(yǔ)言的重載不是太一樣,雖然都實(shí)現(xiàn)了相同的功能。
屬性重載涉及到的兩個(gè)魔術(shù)方法主要是用來(lái)處理屬性訪問(wèn),定義了當(dāng)我們嘗試訪問(wèn)一個(gè)不存在(或不可訪問(wèn))的屬性時(shí)會(huì)發(fā)生什么。

__get()

魔術(shù)方法__get()在我們嘗試訪問(wèn)一個(gè)不存在的屬性時(shí)會(huì)被調(diào)用。它接收一個(gè)參數(shù),該參數(shù)表示訪問(wèn)屬性的名字,并且將該屬性的值返回。在上面的Device類里,有一個(gè)data屬性,該屬性就在這里就起了作用,如下面得代碼:

復(fù)制代碼 代碼如下:

class Device {
    public function  __get($name) {
         if(array_key_exists($name, $this->data)) {
            return $this->data[$name];
        }
        return null;
    }
}

該魔術(shù)方法最常用的地方就是通過(guò)創(chuàng)建一個(gè)“只讀”的屬性來(lái)擴(kuò)展訪問(wèn)控制。在上面的Battery類中,有一個(gè)私有屬性$charge,我們可以通過(guò)__get()魔術(shù)方法將該屬性擴(kuò)展為在類外部可讀但不能修改。代碼如下:

復(fù)制代碼 代碼如下:

class Battery {
    private $charge = 0;
 
    public function  __get($name) {
        if(isset($this->$name)) {
            return $this->$name;
        }
        return null;
    }
}

__set()

__set()魔術(shù)方法在我們嘗試修改一個(gè)不可訪問(wèn)的屬性時(shí)會(huì)被調(diào)用,它接收兩個(gè)參數(shù),一個(gè)表示屬性的名字,一個(gè)表示屬性的值。示例代碼如下:

復(fù)制代碼 代碼如下:

class Device {
    public function  __set($name, $value) {
        // use the property name as the array key
        $this->data[$name] = $value;
    }
}

__isset()

__isset()魔術(shù)方法在對(duì)一個(gè)不可訪問(wèn)的屬性調(diào)用isset()方法時(shí)會(huì)被調(diào)用,它接收一個(gè)參數(shù),表示屬性的名字。它應(yīng)該返回一個(gè)布爾值,用來(lái)表示該屬性是否存在。代碼如下:

復(fù)制代碼 代碼如下:

class Device {
    public function  __isset($name) {
        return array_key_exists($name, $this->data);
    }
}

__unset()

__unset()魔術(shù)方法在調(diào)用unset()函數(shù)銷毀一個(gè)不能訪問(wèn)的屬性時(shí)會(huì)被調(diào)用,它接收一個(gè)參數(shù),表述屬性的名字。

對(duì)象轉(zhuǎn)換為字符串

有時(shí)候我們需要將對(duì)象以字符串的形式表現(xiàn)出來(lái)。如果我們直接打印一個(gè)對(duì)象,那么程序?qū)?huì)輸出一個(gè)錯(cuò)誤信息:PHP Catchable fatal error: Object of class Device could not be converted to string

__toString()

__toString()在我們將對(duì)象當(dāng)作字符串一樣使用時(shí)會(huì)被調(diào)用,它不接收任何參數(shù)。該方法允許我們定義對(duì)象的表現(xiàn)形式。代碼如下:

復(fù)制代碼 代碼如下:

class Device {
    public function  __toString() {
       $connected = (isset($this->connection)) ? 'connected' : 'disconnected';
       $count = count($this->data);
       return $this->name . ' is ' . $connected . ' with ' . $count . ' items in memory' . PHP_EOL;
    }
    ...
}

__set_state()(PHP 5.1)

靜態(tài)魔術(shù)方法__set_state(),在我們使用var_export()函數(shù)輸出對(duì)象時(shí)會(huì)調(diào)用該方法。var_export()函數(shù)用來(lái)將PHP變量轉(zhuǎn)換為PHP代碼,它接收一個(gè)包含對(duì)象屬性值的關(guān)聯(lián)數(shù)組作為參數(shù)。示例代碼如下:

復(fù)制代碼 代碼如下:

class Battery {
    //...
    public static function  __set_state(array $array) {
        $obj = new self();
        $obj->setCharge($array['charge']);
        return $obj;
    }
    //...
}

克隆對(duì)象

默認(rèn)的,對(duì)象都是按引用傳值的。因此,在將一個(gè)對(duì)象賦值給另一個(gè)變量時(shí),只是創(chuàng)建了指向該對(duì)象的一個(gè)引用,并沒(méi)有復(fù)制該對(duì)象。為了實(shí)現(xiàn)真正得復(fù)制一個(gè)對(duì)象,我們需要使用clone關(guān)鍵字。
這種“按引用傳遞”的策略同樣適用于包含在對(duì)象內(nèi)部的對(duì)象。即使我們克隆了一個(gè)對(duì)象,在對(duì)象內(nèi)部的任何對(duì)象都不會(huì)被克隆,因此最終的結(jié)果是兩個(gè)對(duì)象共享了同一個(gè)內(nèi)部對(duì)象。示例代碼如下:

復(fù)制代碼 代碼如下:

$device = new Device(new Battery(), 'iMagic');
$device2 = clone $device;
 
$device->battery->setCharge(65);
echo $device2->battery->charge;
// 65

__clone()

__clone()魔術(shù)方法__clone()可以解決上面的問(wèn)題。當(dāng)對(duì)一個(gè)對(duì)象使用clone關(guān)鍵字時(shí),該魔術(shù)方法會(huì)被調(diào)用。在這個(gè)魔術(shù)方法里,我們可以實(shí)現(xiàn)任何子對(duì)象的克隆,代碼如下:

復(fù)制代碼 代碼如下:

class Device {
    ...
    public function  __clone() {
        // copy our Battery object
        $this->battery = clone $this->battery;
    }
    ...
}

對(duì)象序列化

序列化是講任意數(shù)據(jù)轉(zhuǎn)換為字符串格式的過(guò)程。序列化通常用來(lái)將整個(gè)對(duì)象存入數(shù)據(jù)庫(kù)或?qū)懭胛募?。?dāng)反序列化存儲(chǔ)的數(shù)據(jù)時(shí),我們可以得到序列化之前的對(duì)象。但是,并不是所有得數(shù)據(jù)都可以被序列化,比如數(shù)據(jù)庫(kù)連接。幸運(yùn)的是,有一個(gè)魔術(shù)方法可以幫我們解決這個(gè)問(wèn)題。

__sleep()

魔術(shù)方法__sleep()在對(duì)一個(gè)對(duì)象序列化時(shí)(調(diào)用serialize())會(huì)被調(diào)用。它不接收任何參數(shù),而且應(yīng)該返回一個(gè)包含所有應(yīng)該被序列化的屬性的數(shù)組。在該魔術(shù)方法中,也可以執(zhí)行一些其他操作。
有一點(diǎn)要注意的是,不要再該函數(shù)中進(jìn)行任何的析構(gòu)操作,因?yàn)檫@可能會(huì)影響正在運(yùn)行的對(duì)象。

示例代碼如下:

復(fù)制代碼 代碼如下:

class Device {
    public $name;          
    public $battery;      
    public $data = array();
    public $connection;   
    //...
    public function  __sleep() {
        return array('name', 'battery', 'data');
    }
    //...
}

__wakeup()

魔術(shù)方法__wakeup()在對(duì)存儲(chǔ)的對(duì)象反序列化時(shí)會(huì)被調(diào)用。它不接收任何參數(shù),也沒(méi)有任何返回值。可以用它來(lái)處理在序列化時(shí)丟失的數(shù)據(jù)庫(kù)連接或資源。代碼如下:

復(fù)制代碼 代碼如下:

class Device {
    //...
    public function  __wakeup() {
        // reconnect to the network
        $this->connect();
    }
    //...
}

方法重載

PHP還有兩個(gè)與成員方法相關(guān)的魔術(shù)方法__call()和__callStatic(),這兩個(gè)魔術(shù)方法類似于屬性重載方法。

__call()

魔術(shù)方法__call()在調(diào)用不存在或不可訪問(wèn)的方法時(shí)會(huì)被調(diào)用。它接收兩個(gè)參數(shù),一個(gè)是調(diào)用的方法的名字,一個(gè)是包含函數(shù)參數(shù)的數(shù)組。我們可以使用這種方法調(diào)用子對(duì)象中得同名函數(shù)。

在這個(gè)例子中,要注意函數(shù)call_user_func_array(),這個(gè)函數(shù)允許我們動(dòng)態(tài)調(diào)用一個(gè)命名的函數(shù)。

示例代碼如下:

復(fù)制代碼 代碼如下:

class Device {
    //...
    public function  __call($name, $arguments) {
        // make sure our child object has this method
        if(method_exists($this->connection, $name)) {
            // forward the call to our child object
            return call_user_func_array(array($this->connection, $name), $arguments);
        }
        return null;
    }
    //...
}

__callStatic()

魔術(shù)方法__callStatic()與__call()的功能一樣,唯一不同的是,該方法在嘗試訪問(wèn)一個(gè)不存在或不可訪問(wèn)的靜態(tài)方法時(shí)會(huì)被調(diào)用。示例代碼如下:

復(fù)制代碼 代碼如下:

class Device {
    //...
    public static function  __callStatic($name, $arguments) {
        // make sure our class has this method
        if(method_exists('Connection', $name)) {
            // forward the static call to our class
            return call_user_func_array(array('Connection', $name), $arguments);
        }
        return null;
    }
    //...
}

將對(duì)象作為函數(shù)

有時(shí)候我們會(huì)需要將對(duì)象作為函數(shù)使用。將對(duì)象作為函數(shù)使用,就像我們使用其他普通的函數(shù)一樣,可以傳參。

__invoke()(PHP 5.3)

魔術(shù)方法__invoke()在嘗試將對(duì)象作為函數(shù)使用時(shí)會(huì)被調(diào)用。在該方法中定義的任何參數(shù),都將被作為函數(shù)的參數(shù)。示例代碼如下:

復(fù)制代碼 代碼如下:

class Device {
    //...
    public function __invoke($data) {
        echo $data;
    }
    //...
}
$device = new Device(new Battery(), 'iMagic');
$device('test');
// equiv to $device->__invoke('test')
// Outputs: test

其他:__autoload()

__autoload()方法并不是一個(gè)魔術(shù)方法,但是這個(gè)方法非常有用。但是,對(duì)著PHP版本的更新,該函數(shù)已經(jīng)不建議使用,取而代之的是spl_auto_register()函數(shù)。

相關(guān)文章

最新評(píng)論