串行化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還沒有被賦值
所以,一定要小心你定義類中變量的執行順序。
原文地址:http://blog.sina.com.cn/s/blog_758ddcb90100yk05.html