首頁 >後端開發 >php教程 >PHP 魔術方法: __sleep __wakeup

PHP 魔術方法: __sleep __wakeup

WBOY
WBOY原創
2016-08-08 09:33:091624瀏覽

從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教程有興趣的朋友有所幫助。

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn