串行化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
