PHP串行化與JSON

[轉自]

03月19
作者:hanguofeng[小韓]
發佈於:經典論壇 玄貓的窩(作者的博客)
本文遵從創作共用協議(by-nc-sa)

What 、Why、How

What

Ok,各位親愛的朋友,讓我們開始這個新概念的旅程,串行化這個話題可能大家以前都沒有多加關注,事情其實起源於那天我隨便翻翻PHP手冊,發現這個串行化的函數,之後閒來無聊又做一個WordPress的插件,這個時候順便用了一下串行化,發現在某些場合的確非常方便。
先來解釋下串行化:簡單來說,串行化即將變量轉換成字節流的過程。串行化的提出,有效的解決了對象的保存和傳輸的問題,舉例來說,我在JavaScript中建立了一個對象,我現在想將這個對象保存到服務器端的數據庫中,那麼我如何進行操作呢,這個時候往往就用到了對象的串行化。在JavaScript的串行化中不得不提JSON,JSON(JavaScript
Object Notation) 是一種輕量級的數據交換格式。易於人閱讀和編寫,同時也易於機器解析和生成。它基於JavaScript Programming Language, Standard ECMA-262 3rd Edition - December
1999的一個子集。 JSON採用完全獨立於語言的文本格式,但是也使用了類似於C語言家族的習慣(包括C, C++, C#, Java, JavaScript, Perl, Python等)。這些特性使JSON成爲理想的數據交換語言。
人們通常將JSON和XML進行比較,二者都是將對象扁平化(稍後我們解釋這個”扁平化”)的一種手段,XML的特點是結構嚴謹,而JSON的特點則是簡單易讀、容易使用程序進行分析,因爲它能夠很簡單的將一個對象轉換爲一個字符流的形式,例如如下代碼:
  1. {"type":"human","name":"hanguofeng","age":22}
則是一個JSON表達式,他保存了一個對象,我們如何將它恢復爲對象呢?很簡單,如下:
  1. var animal_str = '{"type":"human","name":"hanguofeng","age":22}'
  2.  
  3. 我們通過JavaScript的求值函數,將JSON表達式進行運算,並返回值,用以獲得一個對象,到這裏,我想你一定會和我一樣,對JSON格式的創造者的思維佩服不已吧。
  4.  
  5. 本來說講串行化的,"不小心"談到JSON,並且講了這麼多,呵呵,跑題了嗎?沒有,PHP的串行化和JSON是非常像的,一個PHP的串行化表達式如下:
  1. a:3:{s:4:"type";s:5:"human";s:4:"name";s:10:"hanguofeng";s:3:"age";s:2:"20";}
他看起來結構和JSON有些類似,實際上,這個表達式是如下數組的串行化結果:
  1. $animal =
  2. (
  3.  
  4. "type" => "human",
  5.  
  6. "name" => "hanguofeng",
  7.  
  8. "age" => "20"
  9.  
  10. );
OK,上面的一些介紹只是讓你大致看到串行化和JSON是什麼樣的東西,你無須對這裏的代碼過分糾結,我們在後面會詳細講解的,下面我們來談談爲什麼要使用串行化。

Why

串行化首先是作爲數據傳輸的方便而出現的,正如本文開始我提出的問題,我在JavaScript中建立了一個對象,我現在想將這個對象保存到服務器端的數據庫中,應該如何做,這其實上是一個”我如何將一個對象從瀏覽器提交到服務器”的問題,在這個傳輸過程中,我們知道,實際上只能夠傳遞字符流,字符流是一維(扁平)的,然而很多對象卻是多維的,如果要傳遞的對象是一個字符串,那麼很簡單,我們直接將其作爲傳遞的內容就可以了,如果要傳遞的對象是一個數組或者其他的結構呢,我們就需要用字符流來描述他,就比如在電話裏面,我問你的名字是什麼,你會告訴我,你的名字是張三、李四,而我問你,你的長相如何呢,你就需要用文字向我描述了,我們進行數據傳遞的媒介往往和這條電話線路一樣,只能傳遞字符流,而我們描述對象的過程,實際上就是串行化的過程。
另外,串行化也可以用於對對象的持久化存儲,也許你曾經也和我一樣,想着在數據庫的某一個字段中存儲一個對象,現在我們可以非常簡單的做到這一點,並且,你的這個數據庫字段不需要設定爲特殊格式,設定爲varchar就可以了(當然,如果對象很大,你可能需要設定爲text)。

How

PHP串行化語法

好了,我想What和Why的問題你都瞭解了,本節最後我們來講點理論性強一些的內容,就是如何使用PHP串行化和反串行化數據,如何將JavaScript對象串行化(即變爲JSON格式)和如何將其反串行化,最後則是如何將JSON和PHP的串行化建立關係。
PHP爲我們提供了兩個函數,用來進行串行化和反串行化的操作,這兩個函數分別是:serialize()和unserialize(),他們適用於PHP4和PHP5,下面分別進行講解:
serialize()
(PHP 4, PHP 5, PECL axis2:0.1.0-0.1.1)
serialize - 獲得一個可存儲的表述值
說明
string serialize ( mixed $value )
獲得一個可存儲的表述值
本函數用於無損的存儲或者傳遞PHP變量值和結構。
如果需要將已經串行化的值轉回PHP變量,可以使用unserialize()函數。
參數
value
即被串行化的表達式。serialize()處理除資源指針之外的所有類型,你甚至可以將含有指向自身元素的數組串行化。你串行化的含有循環指向的數組或者對象一樣會被存儲,其他的指向則會丟失。
當串行化對象時,PHP會嘗試首先調用其成員函數__sleep()。這將允許對象在被串行化之前進行諸如最後的清理工作等。同樣地,當使用unserialize()函數將對象恢復時,會調用成員函數__wakeup()。
返回值
返回一個可以被存儲在任何地點的包含對象的字節流表達式的字符串。
unserialize()
(PHP 4, PHP 5, PECL axis2:0.1.0-0.1.1)
unserialize - 從一個已存儲的表達式中獲得一個PHP變量值
說明
mixed unserialize ( string $str )
unserialize()獲取一個簡單類型的串行化變量並將其轉換回PHP變量值。
參數
str
串行化後的字符串
如果被反串行化的變量是一個對象,則成功恢復該對象的結構後,PHP將自動嘗試執行該對象的__wakeup()成員函數(如果其存在)。
unserialize_callback_func指令:你可以設定在此過程中唄執行的回調函數,如果某個未被定義的類應當在反串行化時被實例化(以避免獲得一個不完全的對象”__PHP_Incomplete_Class”)。你可以使用php.ini,ini_set()或者.htaccess來定義”unserialize_callback_func”。當一個未被定義的類被實例化時,它會被調用。屏蔽這個特性只需將其設爲空即可。
返回值
返回轉換後的數值,可能是布爾變量、實數、浮點數、字符串、數組或者對象。
假如傳入的字符串不可以被反串行化,則返回FALSE,同時拋出NOTICE錯誤。
(以上譯自PHP手冊)

PHP串行化實例

數組的串行化和反串行化
OK,讓我們來用實例學習一下,首先,請建立sample1.php文件,我們在這個文件中用如下語句來創建一個哈希數組:
  1. $animal =
  2. (
  3. "type" => "human",
  4.  
  5. "name" => "hanguofeng",
  6.  
  7. "age" => "20"
  8.  
  9. );
爲了測試這個數組的值,你可以使用print_r()函數來輸出數組,輸出的結果如下:

Array
(
[type] => human
[name] => hanguofeng
[age] => 20
)
 
 
那麼我們將他來串行化一下,串行化的代碼如下:

$animal =
array
(
"type" => "human",
"name" => "hanguofeng",
"age" => "20"
);
$animal_ser=serialize($animal);
echo($animal_ser);
 
 
這裏我們將數組$animal串行化,將返回的串行化字符串保存在變量$animal_ser中,並輸出,輸出的結果是:

a:3:{s:4:"type";s:5:"human";s:4:"name";s:10:"hanguofeng";s:3:"age";s:2:"20";}
 
 
我們來簡單對這個字符串進行一個解析:
a:3表示這是一個數組型的對象(a),他共有三個內置的對象(3)
大括號裏面的部分是以逗號分割的對象表達式列表,以s:4:”type”爲例,他表示一個字符串(s),長度爲4位(4),值爲”type”,即哈希數組的第一個元素的鍵。
後面的部分以此類推,我們不再贅述,你可以試試自己將各種對象串行化,看看串行化後的字符串是如何構建的。
下面來看數組的反串行化,即將我們上面生成的串行化字符串恢復爲數組,代碼如下:

$animal_ser='a:3:{s:4:"type";s:5:"human";s:4:"name";s:10:"hanguofeng";s:3:"age";s:2:"20";}';
$animal = unserialize($animal_ser);
print_r($animal);
 
 
在第一行中,我們假設$animal_ser的值爲上面獲得的串行化字符串,在第二行將該字符串恢復爲開始的數組,並賦值給$animal,最後輸出$animal這個數組,此時的輸出和本節開始時輸出的原始數組是一樣的,即:

Array
(
[type] => human
[name] => hanguofeng
[age] => 20
)
 
 
這樣我們就完成了數組的反串行化。
拓展知識-自定義對象的串行化和反串行化
對數組進行串行化是一個基礎操作,然而在實際的程序設計中,我們可能經常對其他類型的變量進行串行化,例如對某個自定義對象進行串行化,這裏有一個我們自己編寫的類A(保存在classa.inc中):

class A {
var $one = 1;
 
function show_one() {
echo $this->one;
}
}
我們在如下代碼中創建類的實例並對該實例進行串行化:

//省略引入classa.inc文件的代碼
$a=new A;
echo(serialize($a));
 
 
此時輸出的內容爲:

O:1:"A":1:{s:3:"one";i:1;}
 
 
總體來看,這個串行化字符串輸出了改對象當前的狀態,即i的值爲1。下面我們來逐個分析其中的細節。
O:1:由於當前的變量是一個自定義對象,因此該表徵字符爲”O”,表示Object。
後面的”A”標識了該變量是哪個類的實例,這裏即A類。
大括號內即該實例的各個屬性的名稱和值。
而後我們將其進行反串行化:

//省略引入classa.inc文件的代碼
include("classa.inc");
$s = 'O:1:"A":1:{s:3:"one";i:1;}';
$a = unserialize($s);
$a->show_one();
 
 
此時輸出”1″,即調用了A類的show_one()方法。
你可以注意到雖然在實例的串行化字符串中並沒有包含類的方法,但是我們將其反串行化後,仍然可以調用類的方法,這個特性在PHP4及以上版本中被支持(當然,你需要包含類的定義文件classa.inc)。
注:你可以參考PHP手冊中Language Reference->Classes and Objects->Serializing objects - objects in sessions一節的內容。

在JavaScript中串行化爲JSON-使用json2.js

JavaScript中沒有直接串行化對象的內置方法,當然你可以自己寫一個,不過我還是強烈推薦你在這裏偷個小懶,使用現成的組件,JSON的官方網站 “[url]http://www.json.org[/url]”>[url]www.json.org[/url]提供了對JavaScript對象實現JSON串行化的代碼庫-json2.js,你可以從這裏獲得它。
獲得完畢json2.js文件後,你可以打開這個文件,在文件的前部分包含了相當大量的註釋信息,如果你的英文足夠好,那麼你可以省略我這一節,參考該文件的註釋就可以了,如果作爲程序員,你已經看夠了大片的字母,想看看我的漢字+字母,那你可以向下繼續了。
簡單的翻譯下這個註釋:
該文件創建了一個包含兩個方法的全局對象JSON,它的方法分別是:

JSON.stringify(value, whitelist)
 
 
value 任意的JavaScript值,一般是一個對象或者數組
whitelist 一個可選的數組參數,用於判定對象值如何被串行化
這個方法通過一個JavaScript值來生成JSON文本。在進行串行化時,根據可選的參數whitelist,有三種可能:
如果某個對象有toJSON方法,那麼則調用該方法,toJSON方法的返回值將被串行化。
否則,如果可選參數whitelist是一個數組,那麼數組中的元素將被用來選擇對象進行串行化時的的成員。
否則,如果沒有使用whitelist參數,則對象的所有成員將被串行化。
如果值沒有JSON的表現形式,例如undefined或者函數,則其不會被串行化。在對象中,這樣的值會被忽略,而在數組中將會被null替換。
JSON.stringify(undefined)會返回undefined。日期將會被串行化爲被引用的ISO日期。
例:

var text = JSON.stringify(['e', {pluribus: 'unum'}]);
//text is '["e",{"pluribus":"unum"}]‘
 JSON.parse(text, filter)
 
 
該方法解析一個JSON文本,並生成一個組件或者數組,其可能拋出一個SyntaxError異常。
可選的filter參數是一個可過濾和轉換結果的函數、它接受每個鍵和值,它的返回值用來替換源值。如果它返回所接收的值,那麼結果不會被改變。如果他返回undefined,則該成員會被刪除。
例:

//解析文本,如果某個鍵包含字符串"date",則將其值轉換爲日期
myData = JSON.parse(text, function (key, value) {
return key.indexOf('date') >= 0 ? new Date(value) : value;
});
 
 
上面的入門教程已經使你基本瞭解了json2.js的使用方法,這裏關於該文件我就不再贅述了,只是有一個小提示,如果你想簡單的解析一個JSON文本,那麼可以使用eval()函數,改函數是JavaScript的內置函數,例如解析在JSON.stringify的案例中生成的JSON文本,可以使用:

var myE = eval('["e",{"pluribus":"unum"}]‘);
 
 
來獲得對象myE。

在JavaScript中串行化爲JSON-使用prototype.js

如果你和我一樣,喜歡在自己的項目中使用開源的JavaScript框架,那麼你可能可以省去使用json2.js文件了,這裏以protype.js爲例,該文件可以在 “[url]http://www.prototypejs.org[/url]”>[url]http://www.prototypejs.org[/url]下載,由於本文不是在講JavaScript框架,這裏我假設你對prototype.js的使用已經有所瞭解了。
prototype.js中提供了對Object對象的toJSON方法,你可以使用Object.toJSON()方法來實現對對象的串行化,例如:

var cat=
{
name:"hellokitty",
height:"6 apples"
}
alert(Object.toJSON(cat));
//將彈出對話框,內容爲 {"name": "hellokitty", "height": "6 apples"}
 
 
另外,在prototype.js中還有另外的JSON支持,主要是在Ajax對象中對Ajax返回請求中JSON內容的解析。這裏暫時與我們的內容無關,也不再介紹了。

PHP與JSON

在上面我們一起了解了PHP進行對象串行化的方法以及在JavaScript中進行將對象串行化爲JSON的方法,你大致會質疑我爲什麼將二者放在一起,因爲他們的語法實際是不完全一樣的,然而,在PHP中,可以對JSON文本進行反串行化,也可以將PHP的對象串行化爲JSON而非PHP風格的文本。這主要是靠json_decode和json_encode兩個函數來完成的,需要特別說明的是,這兩個函數在PHP
5 >= 5.2.0中才被支持,如果你要編寫運行在PHP4環境下的程序,那麼這兩個函數是不可以使用的。

json_decode函數

語法
mixed json_decode ( string $json [, bool $assoc] )
獲取一個JSON編碼文本,並且將其轉換爲PHP變量
參數
json
被JSON編碼的文本
assoc
當爲TRUE時,返回的值爲聯合數組
返回值
返回一個對象,或者如果可選的assoc參數爲TRUE,則一個聯合數組將會被返回

json_encode函數

語法
string json_encode ( mixed $value )
該函數返回一個值的JSON表達式
參數
value
要被編碼的值,可以爲除resource外的任何類型參數
這個函數僅在UTF-8編碼格式時起作用
返回值
當成功時返回編碼後的JSON文本

json_decode函數實例

下面兩個例子都基於我們的一個情景假設,即,我們有一個用戶註冊的模塊,這個模塊以”面向對象”的方式工作,在json_decode函數實例中,我們在前臺將用戶的註冊信息變爲一個類的屬性,而後傳遞到後臺的php文件(這裏爲了簡便,就不用Ajax了)。在json_encode實例中,我們在html文件中引用一個js文件,地址指向php文件,在php文件中輸出json編碼後的用戶對象(同樣爲了簡便,我們直接生成一個對象而不從數據庫中取信息),並在html中輸出。
好了,先來看前臺的頁面json_encode.htm,這個頁面模仿了通常的註冊頁面,在其上面有一個表單,當提交時,觸發JavaScript函數,生成一個用戶對象user,將表單內容設爲用戶對象的屬性,生成JSON文本,以POST方式傳遞到後臺的json_encode.php文件。在js_encode.php文件中,將JSON文本用json_decode函數解析爲PHP對象,並輸出。
好了,先來看json_encode.html文件,文件代碼如下:

……省略HTML頭部,以下代碼位於head標籤
function JSON_test(o)
{
var user = {
name:document.getElementById('txt_name').value,
email:document.getElementById('txt_email').value,
password:document.getElementById('txt_name').value
}
var json_string = JSON.stringify(user);
 
document.getElementById(’txt_json’).value=json_string;
alert(”點擊確定後將提交表單”);
o.submit();
}
……以下代碼位於body標籤
<form id=”form1″ name=”form1″ method=”post” action=”json_encode.php”
onsubmit=”JSON_test(this)”>
<label for=”txt_name”>姓名</label>
<p>
<input type=”text” name=”txt_name” id=”txt_name” />
</p>
<label for=”txt_email”>郵箱</label>
<p>
<input type=”text” name=”txt_email” id=”txt_email” />
</p>
<p>
<label for=”txt_password”>密碼</label>
</p>
<p>
<input type=”text” name=”txt_password” id=”txt_password” />
</p>
<p>
<input type=”text” name=”txt_json” id=”txt_json” />
<label for=”button”></label>
<input type=”submit” name=”button” id=”button” value=”JSON” />
</p>
</form>
當提交表單時,將觸發JavaScript函數JSON_text(),該函數首先建立一個JavaScript對象user,將其name、email和password屬性分別設爲對應表單的值,而後使用json2.js文件的JSON.stringify方法將其轉換爲JSON文本json_string,最後設定隱藏域(這裏爲了使你看的清楚,我把這個隱藏域以文本框形式顯示了)txt_json的值爲json_string,並提交表單。
下面到json_encode.php文件,如下:

$json_string = $_POST["txt_json"];
if(ini_get("magic_quotes_gpc")=="1")
{
$json_string=stripslashes($json_string);
}
$user = json_decode($json_string);
 
echo var_dump($user);
在這個文件中,首先得到json_encode.html文件中POST表單域txt_json的值,放入變量$json_string中,而後判斷,如果當前PHP的設定爲magic_quotes_gpc=On,即傳入的雙引號等會被轉義,這樣json_decode函數無法解析,因此我們要將其反轉義化。而後,使用json_decode函數將JSON文本轉換爲對象,保存在$user變量中,最終用echo
var_dump($user);,將該對象dump輸出出來,最終結果是:

object(stdClass)#1 (3) {
["name"]=>
string(10) "hanguofeng"
["email"]=>
string(18) “[email][email protected][/email]”
["password"]=>
string(10) “hanguofeng”
}
 
 

json_encode函數實例

在這個例子中,仍然是由兩部分構成的,即json_enode.html和json_encode.php。在json_decode.html文件中,基本與json_decode.html文件的表單類似,但是不同的是,這次我們要從json_encode.php文件中獲得相應用戶的JSON文本,先來看這個PHP文件吧:

Class user
{
public $name="";
public $email="";
public $password="";
};
$myUser = new user;
$myUser->name="hanguofeng";
$myUser->email="[email][email protected][/email]";
$myUser->password="hanguofeng";
$json_string = json_encode($myUser);
 
var user = echo($json_string);
這個文件首先建立類user,而後獲得一個user類的實例myUser,並設定其用戶名、郵箱和密碼,接下來使用json_encode函數將其轉換爲JSON文本,保存在變量$json_string中,最後輸出一段JavaScript代碼,以在JavaScript中建立全局變量user。
接下,我們需要在json_encode.html文件中引入json_encode.php文件,並得到user對象的各個屬性,在這個文件中,你需要注意兩點,第一是,我們以這樣的代碼引入json_encode.php文件爲JavaScript文件:

<script src="json_encode.php" type="text/javascript"></script>
 
 
第二點則是:
我們在文本框代碼後面加入JavaScript的代碼,對文本框的value屬性進行操作,分別設定其值爲user對象的相應值。

實踐出真知

背景說明

OK,各位親愛的朋友,在經過上面這麼多這麼多的零散的關於PHP的串行化和JavaScript的串行化—JSON的說明後,我們來做一個完整的東西,在這裏面,我們涉及了JavaScript中的對象操作、對象串行化,還涉及了當串行化的JSON文本傳輸到後臺的PHP時,PHP進行的相應處理。
這裏需要說明的是,爲了將內容聚焦於本文所闡述的領域,我們將這個項目精簡了很多,在這裏講的並不完全是實際生產環境下的做法,而是爲了突出串行化的應用,在具體的地方我會詳細說明,希望不影響你對整體的理解。
我們要完成的小案例是一個在線記事本的程序,用戶可以在前臺添加若干個Tab(即通常所說的選項卡),每個Tab保存一定的文本內容,你也可以將它當作一個動態的選項卡生成和管理的程序。
程序的運行界面如下圖:
圖1
在這裏,你可以看到有若干個Tab,當點擊這些Tab時,下面將顯示Tab的內容,同時你可以刪除Tab和新增Tab,在完成對Tab的編輯後,可以點擊“保存”按鈕,將改動傳輸到後臺的PHP進行處理,並保存在服務器上。

前臺JavaScript部分

在前臺的notepat.html文件中,我們要使用JavaScript完成大多數對Tab進行管理的任務,因此我構建了一個Tab管理的類,用來對其進行添加、刪除等操作,這個類如下所示:
圖2
tabManager類的各個屬性、方法含義如下:
_tabs:“私有屬性”(加引號的原因是JavaScript中並沒有私有公有的概念,這裏只是從設計上來講),該屬性中存儲了當前的tab數組,任何對Tab的操作將最終存儲在這裏。
_lastIndex:“私有屬性”,表明最後添加的索引,用於對各個Tab進行唯一標識。
getTabs():返回當前的tab數組。
setTabs(tabs):設定當前的tab數組。
add(ptitle,pcontent):新增一個標題爲ptitle,內容爲pcontent的tab。
del(pid):刪除id爲pid的tab。
clean():“私有方法”,用於刪除某個tab後,對tab數組進行精簡壓縮。
getTab(pid):用於獲得指定的單個tab,參數pid爲要獲得的tab的id。
debug():預留的debug方法。
同時注意我們的tab對象有三個屬性,分別爲id、name和content。
這樣,基礎類的架構我們就講解完畢了,這裏由於其詳細代碼與本文話題關聯不大,就不再詳細列出和講解了,稍後可能會單獨發表文章來說這裏。
那麼,我們還需要構建一些函數,作爲控制器,來溝通頁面上各個按鈕的點擊事件和之前建立的tabManager類,這些函數包括:
selTab(o):當點擊頁面某個Tab時,切換內容區域以顯示相應Tab的內容,傳入參數爲所點擊的連接對象。
newTab():當點擊“新建Tab”按鈕時,負責詢問用戶新增Tab的信息,並調用tabManager.add()方法將用戶輸入的Tab信息建立一個新的Tab對象。
delTab():刪除當前激活的Tab。
saveTab():將對Tab的所有更改提交到PHP中。
loadTab():加載Tab。
showTabContent(id):被selTab()函數調用以顯示指定id的Tab的內容到內容區域。
draw(tabs):根據傳入的tabs數組重繪Tab導航區域。
那麼,這裏我要重點強調的函數就有兩個,分別是saveTab()和loadTab()。
先來看saveTab()函數:此函數源代碼如下:
function saveTab()
{
document.getElementById(”json_string”).value = JSON.stringify(tabManager.getTabs());
document.getElementById(”frmMain”).submit();
}
呵呵,好像比較簡單啊,這個函數首先通過調用tabManager類的getTabs()方法,獲得我們所操作的Tabs數組,而後調用json2.js文件中的JSON.stringfy()方法將其串行化,得到串行化的JSON文本後,將其賦給隱藏域json_string,並提交表單,這裏我採取了簡單的方法,其實你在這裏也可以採用Ajax的方式,將JSON文本傳遞到PHP中。
另外則是函數loadTab(),其代碼如下:
function loadTab()
{
tabManager.setTabs(notes);
draw(tabManager.getTabs());
}
這裏需要說明的是,我們在本文件的開頭,通過如下代碼:
<script type=”text/javascript” src=”load.php”></script>
引入了一個經過後臺處理的(你可以看到,擴展名爲PHP)JavaScript文件load.php,。在這個文件中,建立了一個tab對象的數組notes。因此,在loadTab函數中,我們調用了tabManager類的setTabs方法,設定tabManager類的“私有屬性”_tabs的值爲notes數組,此時就完成了對Tab的加載。

後臺PHP部分

當用戶在網頁上點擊“保存”按鈕後,會觸發前面所講解的saveTab()函數,最終提交表單到save.php頁面,該頁面讀取用戶提交的JSON文本,並將其轉換爲PHP對象,進行後續處理,其代碼如下:

$json_string = $_POST["json_string"];
if(ini_get(”magic_quotes_gpc”)==”1″)
{
$json_string=stripslashes($json_string);
}
$notes = json_decode($json_string);
foreach ($notes as $note)
{
echo $note->title.”<br>”.$note->content.”<br>————————-<br>”;
}
$p = f open(”load.php”,”w+”);
f write($p,”var notes=”.json_encode($notes).”;”);
f close($p);

首先通過訪問全局集合$_POST來獲得提交的json_string隱藏域的值,而後進行處理,如果PHP的配置magic_quotes_gpc爲On,則此時提交的文本中,引號會被轉義,如果不進行處理,那麼後面的json解碼操作會失敗,因此我們調用函數stripslashes()將其反轉義。
接下來我們使用json_decode()函數對提交的JSON文本進行解碼,返回的對象保存在$notes變量中,此時$notes變量爲一個數組,數組中每一個元素即一個Tab對象,爲了清楚的看到這一點,我們對$notes數組進行迭代,依次打印出數組中每一個Tab對象的標題和內容,最後,我們打開文件load.php,將使用json_encode進行JSON編碼後的$notes對象寫入此文件,以待notepad.html文件調用,這樣就完成了對對象的持久化存儲工作。
這裏仍然需要說明的是我們精簡的部分:在實際的生產環境中,你很可能需要在迭代$notes數組時將每一個Tab對象的信息寫入數據庫中,以方便其他的部分調用,而在load.php文件中,你可能並不是直接將JSON文本寫入此文件,而是從數據庫中讀取一個結果集,而後迭代結果集,來生成一個PHP的數組,並將PHP數組進行JSON編碼後,輸出。
好了,絮絮叨叨一大堆,簡單點用個圖來表示吧:
圖3
圖4

我還有話要說

各位親愛的~讀者朋友,我們這篇“長”達一萬五千字的連載完成了,在本文中,我想你瞭解了關於對象串行化的一些基本概念以及串行化在PHP和JavaScript中的使用方法。
串行化只是一種思路,他的妙用在於你熟悉了在各類語言中對對象的操作之後,JSON和XML各有特點,根據自己的情況進行優選才是王道。
歡迎大家在論壇或者博客中與我交流。
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章