目錄
序列化
序列化物件##從上面的範例我們可以看出序列化物件的時候,只會保存屬性值。
PHP预定义序列化接口Serializable
题外话
首頁 後端開發 php教程 深入了解PHP中的序列化與反序列化

深入了解PHP中的序列化與反序列化

Feb 26, 2021 pm 06:11 PM
php 反序列化 序列化

本篇文章帶大家深度剖析一下PHP中的序列化與反序列化。有一定的參考價值,有需要的朋友可以參考一下,希望對大家有幫助。

深入了解PHP中的序列化與反序列化

【推薦學習:《PHP影片教學》】

序列化

#序列化格式

在PHP中,序列化用於儲存或傳遞PHP 的值的過程中,同時不遺失其類型和結構。

序列化函數原型如下:

string serialize ( mixed $value )
登入後複製

先看下面的範例:

class CC {
	public $data;
	private $pass;

	public function __construct($data, $pass)
	{
		$this->data = $data;
		$this->pass = $pass;
	}
}
$number = 34;
$str = 'uusama';
$bool = true;
$null = NULL;
$arr = array('a' => 1, 'b' => 2);
$cc = new CC('uu', true);

var_dump(serialize($number));
var_dump(serialize($str));
var_dump(serialize($bool));
var_dump(serialize($null));
var_dump(serialize($arr));
var_dump(serialize($cc));
登入後複製

輸出結果為:

string(5) "i:34;"
string(13) "s:6:"uusama";"
string(4) "b:1;"
string(2) "N;"
string(30) "a:2:{s:1:"a";i:1;s:1:"b";i:2;}"
string(52) "O:2:"CC":2:{s:4:"data";s:2:"uu";s:8:" CC pass";b:1;}"
登入後複製

所以序列化對於不同型別得到的字串格式為:

  • String : s:size:value;
  • Integer : i:value;
  • Boolean : b:value;(保存1或0)
  • #Null : N;
  • Array# : a:size:{key definition;value definition;(repeated per element)}
  • ##Object : O:strlen(object name):object name:object size:{s:strlen (property name):property name:property definition;(repeated per property)}

序列化物件##從上面的範例我們可以看出序列化物件的時候,只會保存屬性值。

那麼物件中的常數會不會保存呢?
  • 如果是繼承,父類別的變數會不會保存呢
  • class CB {
    	public $CB_data = 'cb';
    }
    
    class CC extends CB{
    	const SECOND = 60;
    
    	public $data;
    	private $pass;
    
    	public function __construct($data, $pass)
    	{
    		$this->data = $data;
    		$this->pass = $pass;
    	}
    
    	public function setPass($pass)
    	{
    		$this->pass = $pass;
    	}
    }
    $cc = new CC('uu', true);
    
    var_dump(serialize($cc));
    登入後複製
  • 輸出結果為:
string(75) "O:2:"CC":3:{s:4:"data";s:2:"uu";s:8:" CC pass";b:1;s:7:"CB_data";s:2:"cb";}"
登入後複製

顯然,當序列化物件時,不會儲存常量的值。對於父類別中的變量,則會保留。

物件序列化自訂

在序列化物件的時候,對於物件中的一些敏感屬性,我們不需要保存,這又該如何處理呢?

當呼叫

serialize()

函數序列化物件時,函數會檢查類別中是否存在一個魔術方法__sleep()。如果存在,該方法會先被調用,然後再執行序列化操作。可以透過重載這個方法,從而自訂序列化行為。這個方法原型如下:<div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:php;toolbar:false">public array __sleep ( void )</pre><div class="contentsignin">登入後複製</div></div>

該方法傳回一個包含物件中所有應被序列化的變數名稱的陣列
  • #該方法未傳回任何內容,則NULL 會被序列化,並產生一個
  • E_NOTICE
  • 等級的錯誤
  • __sleep()
  • 不能傳回父類別的私有成員的名字。這樣做會產生一個E_NOTICE等級的錯誤。這時只能用Serializable介面來替代。 常用於保存那些大物件時的清理工作,避免保存過多冗餘資料
  • #看下面的範例:
class User{
	const SITE = 'uusama';

	public $username;
	public $nickname;
	private $password;

	public function __construct($username, $nickname, $password)
	{
		$this->username = $username;
		$this->nickname = $nickname;
		$this->password = $password;
	}

	// 重载序列化调用的方法
	public function __sleep()
	{
		// 返回需要序列化的变量名,过滤掉password变量
		return array('username', 'nickname');
	}
}
$user = new User('uusama', 'uu', '123456');
var_dump(serialize($user));
登入後複製

回傳結果如下,顯然序列化的時候忽略了password 欄位的值。

string(67) "O:4:"User":2:{s:8:"username";s:6:"uusama";s:8:"nickname";s:2:"uu";}"
登入後複製

序列化物件儲存

透過上面的介紹,我們可以把一個複製的物件或資料序列化成一個序列字串,保存值的同事還保存了他們的結構。

我們可以把序列化之後的值存起來,存在檔案或快取裡面。不建議存在資料庫裡面,可讀性查,而且不便於遷移維護,不便查詢。

$user = new User('uusama', 'uu', '123456');
$ser = serialize($user);
// 保存在本地
file_put_contents('user.ser', $ser);
登入後複製

反序列化

使用方法

透過上面的講解,我們可以將物件序列化為字串並保存起來,那麼如何把這些序列化後的字串恢復成原樣呢? PHP提供了反序列函數:

mixed unserialize ( string $str )
登入後複製

unserialize()

反序列化函數用於將單一的已序列化的變數轉換回 PHP 的值。

如果傳遞的字串不可解序列化,則傳回FALSE,並產生一個
    E_NOTICE
  • 傳回的是轉換之後的值,可為
  • integer``float
  • stringarrayobject若被反序列化的變數是一個對象,在成功重新建構物件之後,PHP會自動地試圖去呼叫
  • __wakeup()
  • 成員函數(如果存在的話)
  • 看下面的範例:
class User{
	const SITE = 'uusama';

	public $username;
	public $nickname;
	private $password;
	private $order;

	public function __construct($username, $nickname, $password)
	{
		$this->username = $username;
		$this->nickname = $nickname;
		$this->password = $password;
	}

	// 定义反序列化后调用的方法
	public function __wakeup()
	{
		$this->password = $this->username;
	}
}
$user_ser = 'O:4:"User":2:{s:8:"username";s:6:"uusama";s:8:"nickname";s:2:"uu";}';
var_dump(unserialize($user_ser));
登入後複製

輸出結果為:

object(User)#1 (4) {
  ["username"]=>
  string(6) "uusama"
  ["nickname"]=>
  string(2) "uu"
  ["password":"User":private]=>
  string(6) "uusama"
  ["order":"User":private]=>
  NULL
}
登入後複製

可以得到以下結論:

    __wakeup()
  • 函數在物件被建構後執行,所以$this->username的值不為空反序列化時,會盡量將變數值進行配對並複製給序列化後的物件
未定義類別的處理

在上面的範例中,我們在呼叫反序列化函數

unserialize()

之前,事先定義了User類,如果我們沒有定義會怎麼樣呢? <div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:php;toolbar:false">$user_ser = 'O:4:&quot;User&quot;:2:{s:8:&quot;username&quot;;s:6:&quot;uusama&quot;;s:8:&quot;nickname&quot;;s:2:&quot;uu&quot;;}'; var_dump(unserialize($user_ser));</pre><div class="contentsignin">登入後複製</div></div>這個例子中,我們沒有定義任何的

User

類,反序列化正常執行,並沒有報錯,得到的結果如下:<div class="code" style="position:relative; padding:0px; margin:0px;"><pre class="brush:php;toolbar:false">object(__PHP_Incomplete_Class)#1 (3) {   [&quot;__PHP_Incomplete_Class_Name&quot;]=&gt;   string(4) &quot;User&quot;   [&quot;username&quot;]=&gt;   string(6) &quot;uusama&quot;   [&quot;nickname&quot;]=&gt;   string(2) &quot;uu&quot; }</pre><div class="contentsignin">登入後複製</div></div>注意對比之前定義了

User

類別的結果,這兒反序列化得到的物件是__PHP_Incomplete_Class,並指定了未定義類別的類別名稱。 <p>如果这个时候我们去使用这个反序列化后的不明对象,则会抛出<code>E_NOTICE。这么看着不能用也不是办法,那么如何处理呢?有两种方案。

  • 定义__autoload()等函数,指定发现未定义类时加载类的定义文件
  • 可通过 php.ini、ini_set() 或 .htaccess 定义unserialize_callback_func。每次实例化一个未定义类时它都会被调用

以上两种方案的实现如下:

// unserialize_callback_func 从 PHP 4.2.0 起可用
ini_set('unserialize_callback_func', 'mycallback'); // 设置您的回调函数
function mycallback($classname) 
{
   // 只需包含含有类定义的文件
   // $classname 指出需要的是哪一个类
}


// 建议使用下面的函数,代替__autoload()
spl_autoload_register(function ($class_name) {
	// 动态加载未定义类的定义文件
    require_once $class_name . '.php';
});
登入後複製

PHP预定义序列化接口Serializable

还记得上面在将序列化过程中遇到的:无法在__sleep()方法中返回父类对象的问题吗,方法就是实现序列化接口Serializable

该接口的原型如下:

Serializable {
	abstract public string serialize ( void )
	abstract public mixed unserialize ( string $serialized )
}
登入後複製

需要注意的是,如果定义的类实现了Serializable接口,那么序列化和反序列化的时候,PHP就不会再去调用__sleep()方法和__wakeup()方法。

class CB implements Serializable{
	public $CB_data = '';
	private $CB_password = 'ttt';

	public function setCBPassword($password)
	{
		$this->CB_password = $password;
	}

	public function serialize()
	{
		echo __METHOD__ . "\n";
		return serialize($this->CB_password);
	}

	public function unserialize($serialized)
	{
		echo __METHOD__ . "\n";
	}
}

class CC extends CB {
	const SECOND = 60;

	public $data;
	private $pass;

	public function __construct($data, $pass)
	{
		$this->data = $data;
		$this->pass = $pass;
	}

	public function __sleep()
	{
		// 输出调用了该方法名
		echo __METHOD__ . "\n";
	}

	public function __wakeup()
	{
		// 输出调用了该方法名
		echo __METHOD__ . "\n";
	}
}
$cc = new CC('uu', true);
$ser = serialize($cc);
var_dump($ser);
$un_cc = unserialize($ser);
var_dump($un_cc);
登入後複製

运行结果为:

CB::serialize
string(24) "C:2:"CC":10:{s:3:"ttt";}"
CB::unserialize
object(CC)#2 (4) {
  ["data"]=>
  NULL
  ["pass":"CC":private]=>
  NULL
  ["CB_data"]=>
  string(0) ""
  ["CB_password":"CB":private]=>
  string(3) "ttt"
}
登入後複製

可以完全定义serialize()方法,该方法返回的值就是序列化后大括号内的值,只要保证自定义序列化和反序列化的规则一致即可。

题外话

在PHP应用中,序列化和反序列化一般用做缓存,比如session缓存,cookie等。

序列化和反序列化在PHP中用得不算多,在Java语言中用得比较多。其实你有没有发现,这种把一个对象或者数组的变量转化成字符串的方式,json也可以做到。

使用json来实现对象和字符串之间的转换,在PHP中显得更加直观和轻便。而且经过测试,使用json_encode()serialize()方法更加快速,大概快2~3倍。

在我看来,序列化和反序列化是一种传输抽象数据的思想。通过定义序列化和反序列化的规则,我们可以实现将PHP中的对象序列化成字节流,然后传输给别的语言或者系统使用,这在远程调用里面非常的方便。

更多编程相关知识,请访问:编程视频!!

以上是深入了解PHP中的序列化與反序列化的詳細內容。更多資訊請關注PHP中文網其他相關文章!

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

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

AI Hentai Generator

AI Hentai Generator

免費產生 AI 無盡。

熱門文章

R.E.P.O.能量晶體解釋及其做什麼(黃色晶體)
1 個月前 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O.最佳圖形設置
1 個月前 By 尊渡假赌尊渡假赌尊渡假赌
威爾R.E.P.O.有交叉遊戲嗎?
1 個月前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

適用於 Ubuntu 和 Debian 的 PHP 8.4 安裝和升級指南 適用於 Ubuntu 和 Debian 的 PHP 8.4 安裝和升級指南 Dec 24, 2024 pm 04:42 PM

PHP 8.4 帶來了多項新功能、安全性改進和效能改進,同時棄用和刪除了大量功能。 本指南介紹如何在 Ubuntu、Debian 或其衍生版本上安裝 PHP 8.4 或升級到 PHP 8.4

如何設定 Visual Studio Code (VS Code) 進行 PHP 開發 如何設定 Visual Studio Code (VS Code) 進行 PHP 開發 Dec 20, 2024 am 11:31 AM

Visual Studio Code,也稱為 VS Code,是一個免費的原始碼編輯器 - 或整合開發環境 (IDE) - 可用於所有主要作業系統。 VS Code 擁有大量針對多種程式語言的擴展,可以輕鬆編寫

我後悔之前不知道的 7 個 PHP 函數 我後悔之前不知道的 7 個 PHP 函數 Nov 13, 2024 am 09:42 AM

如果您是經驗豐富的PHP 開發人員,您可能會感覺您已經在那裡並且已經完成了。操作

您如何在PHP中解析和處理HTML/XML? 您如何在PHP中解析和處理HTML/XML? Feb 07, 2025 am 11:57 AM

本教程演示瞭如何使用PHP有效地處理XML文檔。 XML(可擴展的標記語言)是一種用於人類可讀性和機器解析的多功能文本標記語言。它通常用於數據存儲

在PHP API中說明JSON Web令牌(JWT)及其用例。 在PHP API中說明JSON Web令牌(JWT)及其用例。 Apr 05, 2025 am 12:04 AM

JWT是一種基於JSON的開放標準,用於在各方之間安全地傳輸信息,主要用於身份驗證和信息交換。 1.JWT由Header、Payload和Signature三部分組成。 2.JWT的工作原理包括生成JWT、驗證JWT和解析Payload三個步驟。 3.在PHP中使用JWT進行身份驗證時,可以生成和驗證JWT,並在高級用法中包含用戶角色和權限信息。 4.常見錯誤包括簽名驗證失敗、令牌過期和Payload過大,調試技巧包括使用調試工具和日誌記錄。 5.性能優化和最佳實踐包括使用合適的簽名算法、合理設置有效期、

php程序在字符串中計數元音 php程序在字符串中計數元音 Feb 07, 2025 pm 12:12 PM

字符串是由字符組成的序列,包括字母、數字和符號。本教程將學習如何使用不同的方法在PHP中計算給定字符串中元音的數量。英語中的元音是a、e、i、o、u,它們可以是大寫或小寫。 什麼是元音? 元音是代表特定語音的字母字符。英語中共有五個元音,包括大寫和小寫: a, e, i, o, u 示例 1 輸入:字符串 = "Tutorialspoint" 輸出:6 解釋 字符串 "Tutorialspoint" 中的元音是 u、o、i、a、o、i。總共有 6 個元

解釋PHP中的晚期靜態綁定(靜態::)。 解釋PHP中的晚期靜態綁定(靜態::)。 Apr 03, 2025 am 12:04 AM

靜態綁定(static::)在PHP中實現晚期靜態綁定(LSB),允許在靜態上下文中引用調用類而非定義類。 1)解析過程在運行時進行,2)在繼承關係中向上查找調用類,3)可能帶來性能開銷。

什麼是PHP魔術方法(__ -construct,__destruct,__call,__get,__ set等)並提供用例? 什麼是PHP魔術方法(__ -construct,__destruct,__call,__get,__ set等)並提供用例? Apr 03, 2025 am 12:03 AM

PHP的魔法方法有哪些? PHP的魔法方法包括:1.\_\_construct,用於初始化對象;2.\_\_destruct,用於清理資源;3.\_\_call,處理不存在的方法調用;4.\_\_get,實現動態屬性訪問;5.\_\_set,實現動態屬性設置。這些方法在特定情況下自動調用,提升代碼的靈活性和效率。

See all articles