從php5以後的版本,類別就可以使用魔術方法了。 php規定以兩個底線(__)開頭的方法都保留為魔術方法,所以建議大家函數名最好不用__開 頭,除非是為了重載已有的魔術方法。
目前php已有的魔術方法有 __construct,__destruct,__call,__get,__set,__isset,__unset,__sleep,__wakeup,__toString,__set_state 和 __clone。
下面說說php魔術方法中的__sleep __wakeup:
串列化serialize可以把變數包括物件,轉換成連續bytes資料. 你可以將串列化後的變數存在一個檔案或在網路上傳輸. 然後再反串列化還原為原來的資料. 你在反串行化類別的物件之前定義的類別,PHP可以成功地儲存其物件的屬性和方法. 有時你可能需要一個物件在反串行化後立即執行. 為了這樣的目的,PHP會自動尋找__sleep和__wakeup方法.
當一個物件被串列化,PHP會呼叫__sleep方法(如果存在的話). 在反串列化一個物件後,PHP 會呼叫__wakeup方法. 這兩個方法都不接受參數. __sleep方法必須回傳一個數組,包含需要串列化的屬性. PHP會拋棄其它屬性的值. 如果沒有__sleep方法,PHP將保存所有屬性.
在程式執行前,serialize() 函數會先檢查是否存在一個魔術方法 __sleep.如果存在,__sleep()方法會先被調用,
然後才執行串行化(序列化)操作。這個函數可以用來清理對象,並傳回一個包含對像中所有變數名稱的陣列。如果該方法不傳回任何內容,則NULL被序列化,導致
一個E_NOTICE錯誤。與之相反,unserialize()會檢查是否有一個__wakeup方法。如果存在,則會先調用
__wakeup方法,預先準備物件資料。
__sleep方法常用於提交未提交的數據,或類似的操作。同時,如果你有一些很大的對象, 不需要保存,這個功能就很好用。 __wakeup經常用在反序列化操作中,例如重新建立資料庫連接,或執行其它初始化操作。
<?php class Connection { protected $link; private $server, $username, $password, $db; public function __construct($server, $username, $password, $db) { $this->server = $server; $this->username = $username; $this->password = $password; $this->db = $db; $this->connect(); } private function connect() { $this->link = mysql_connect($this->server, $this->username, $this->password); mysql_select_db($this->db, $this->link); } public function __sleep() { return array('server', 'username', 'password', 'db'); } public function __wakeup() { $this->connect(); } } ?>
下面範例顯示如何用__sleep和__wakeup方法來串行化一個物件. Id屬性是一個不打算保留在物件中的臨時屬性. __sleep方法保證在串行化的物件中不包含id屬性. 當反串行化一個User對象,__wakeup方法建立id屬性的新值. 這個例子被設計成自我保持. 在實際開發中,你可能發現包含資源(如圖像或數據流)的對象需要這些方法。
<?php class user { public $name; public $id; function __construct() { // 给id成员赋一个uniq id $this->id = uniqid(); } function __sleep() { //此处不串行化id成员 return(array('name')); } function __wakeup() { $this->id = uniqid(); } } $u = new user(); $u->name = "Leo"; $s = serialize($u); //serialize串行化对象u,此处不串行化id属性,id值被抛弃 $u2 = unserialize($s); //unserialize反串行化,id值被重新赋值 //对象u和u2有不同的id赋值 print_r($u); print_r($u2); ?>
例三:__wakeup方法的一個缺陷要注意,如果你打算unserialize一個對象,你
<?php class A { public $b; public $name; } class B extends A { public $parent; public function __wakeup() { var_dump($parent->name); } } $a = new A(); $a->name = "foo"; $a->b = new B(); //我们期望这里输出:foo,但实际在后面的代码执行之后,实际输出NULL. $a->b->parent = $a; $s = serialize($a); $a = unserialize($s); ?>
原因: $b 物件在$name之前unserialized了. 所以在B::__wakeup執行時, $a->name還沒有被賦值
所以,一定要小心你定義類別中變數的執行順序。
以上就介紹了PHP 魔術方法: __sleep __wakeup,包括了PHP 魔術方法方面的內容,希望對PHP教程有興趣的朋友有所幫助。