程序師世界是廣大編程愛好者互助、分享、學習的平台,程序師世界有你更精彩!
首頁
編程語言
C語言|JAVA編程
Python編程
網頁編程
ASP編程|PHP編程
JSP編程
數據庫知識
MYSQL數據庫|SqlServer數據庫
Oracle數據庫|DB2數據庫
 程式師世界 >> 編程語言 >> 網頁編程 >> PHP編程 >> 關於PHP編程 >> PHP 魔術方法 __sleep __wakeup(四),__sleep__wakeup

PHP 魔術方法 __sleep __wakeup(四),__sleep__wakeup

編輯:關於PHP編程

PHP 魔術方法 __sleep __wakeup(四),__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還沒有被賦值

所以,一定要小心你定義類中變量的執行順序。

 

原文地址:http://blog.sina.com.cn/s/blog_758ddcb90100yk05.html

  1. 上一頁:
  2. 下一頁:
Copyright © 程式師世界 All Rights Reserved