談JSON在Ajax中的使用
JSON是一種輕量級的數(shù)據(jù)交換格式。易于人閱讀和編寫,同時也易于機器解析和生成。AJAX是一種用于創(chuàng)建更好更快以及交互性更強的 Web 應(yīng)用程序的技術(shù)。之前也曾介紹過在PHP語言中使用JSON的文章,大家也可以看看,供參考。
雖然XML在Ajax運行中具有舉足輕重的地位,但JavaScript開發(fā)人員很快就對它失去了興趣。在JavaScript中操作XML存在嚴(yán)重的跨瀏覽器問題,而且從XML結(jié)構(gòu)中提取數(shù)據(jù)也需要涉及遍歷DOM文檔,而這些操作都需要編寫大量的代碼。Douglas Crockford發(fā)明了一種叫JSON(JavaScript Object Notation)的數(shù)據(jù)格式卻能夠創(chuàng)建與XML相同的數(shù)據(jù)結(jié)構(gòu)。JSON的基礎(chǔ)是JavaScript語法中一個子集,特別是對象和數(shù)組字面量。JSON的設(shè)計意圖是在服務(wù)器構(gòu)建格式化數(shù)據(jù),然后再將數(shù)據(jù)發(fā)送給瀏覽器。
由于JSON在JavaScript中相當(dāng)于對象和數(shù)組,轉(zhuǎn)換速度快,而且便于在JavaScript代碼中訪問,JSON在Ajax通信中越來越受開發(fā)人員的追捧。Web開發(fā)社區(qū)已經(jīng)為幾乎所有主流的語言都開發(fā)了JSON解析器和序列化器,使得通過服務(wù)器輸出和使用JSON數(shù)據(jù)變得極為容易。
Douglas Crockford自己也維護(hù)著一個針對JavaScript的JSON序列化器/解析器,下載地址為http://www.json.org/js.html,可以去下載那個JavaScript文件,且該文件在所有瀏覽器都能正常使用。此外,IE8中包含了Crockford解析器的原生版本。
在Crockford的這個JSON庫中,有一個全局對象,這個對象有兩個方法:parse()和stringify()。其中,parse()方法接受兩個參數(shù):JSON文本和一個可選的過濾函數(shù)。在傳入的文本是有效的JSON的情況下,parse()方法會返回傳入數(shù)據(jù)的一個對象表示。下面是使用parse()方法的示例:
- var object=JSON.parse("()");
與直接使用eval()不同的是這里不需要為傳入的文本加圓括號。
第二個參數(shù)是一個函數(shù),這個函數(shù)以一個JSON鍵和值作為參數(shù)。要想讓作為參數(shù)傳入的的鍵出現(xiàn)有結(jié)果對象中,該函數(shù)必須返回一個值。它的返回值將成為結(jié)果對象中與指定鍵關(guān)聯(lián)的值,因此也就為我們重寫默認(rèn)的解析機制提供了機會。換句話說,在這個函數(shù)中針對某個鍵返回undefined,就會從結(jié)果對象中移除該鍵。如下面的例子所示:
- var jsontext="{"\name\":\"WangGang\",\"age\":29,\"author\":true }";
- var object=JSON.parse(jsontext,function(key,value){
- switch(key){
- case "age": return value+1;
- case "author": return undefined;
- default: return false;
- }
- });
- alert(object.age) //30
- alert(object.author) //undefined
在以上代碼中,過濾函數(shù)會為每個"age"鍵的值加1,會移除數(shù)據(jù)中的"author"鍵,其他值則會原樣返回。于是,結(jié)果對象中的age屬性值就變成了30,但是卻沒有author屬性,這種解析功能經(jīng)常用于處理服務(wù)器返回的數(shù)據(jù)。假設(shè)addressbook.php會以下面的格式返回JSON數(shù)據(jù):
- {
- {
- "name":"WangMeng",
- "email":"wangmeng@some-domain-name.com"
- },{
- "name":"LinTao",
- "email":"linTao@some-domain-name.com"
- },{
- "name":"Jim",
- "email":"jim@some-domain-name.com"
- }
- }
可以發(fā)送一個Ajax請求取得以上數(shù)據(jù),然后在客戶端使用下列代碼生成相應(yīng)的<ul/>元素:
- var xhr=createXHR();
- xhr.onreadystatechange=function(){
- if(xhr.readyState == 4){
- if((xhr.status >= 200 && xhr.status < 300)|| xhr.status == 304){
- var contacts=JSON.parse(xhr.responseText);
- var list=document.getElementById("contacts");
- for(var i=0,len=contacts.length;i<len;i++){
- var li=document.createElement("li");
- li.innerHTML="<a href=\"mailto:" + contacts[i].email + "\">" + contacts[i].name + "</a>";
- list.appendChild(li);
- }
- }
- xhr.open("get","addressbook.php",true);
- xhr.send(null);
- }
- };
以上代碼從服務(wù)器取得了JSON字符串,然后將它解析成了JavaScript數(shù)組,得到數(shù)組后,通過迭代遍歷其中的每個對象,很容易就可以將相應(yīng)的值插入到DOM中。具體來說,<ul/>元素會包含一些<li/>元素,而每個<li/>元素則會包含一個鏈接,點擊可以向一個人發(fā)送電子郵件。
JSON同樣也是向服務(wù)器發(fā)送數(shù)據(jù)的瀏覽格式。發(fā)送數(shù)據(jù)時,一般會把JSON放到POST元素請求主體中,而JSON對象的stringify()方法正是為此而設(shè)計的。這個方法接受3個參數(shù):要序列化的對象,可選的替換函數(shù)(用于替換未接受的JSON值)和可選的縮進(jìn)說明符(可以是每個級別縮進(jìn)的空格數(shù),也可是用來縮進(jìn)的字符)。默認(rèn)情況下,stringify()返回未經(jīng)縮進(jìn)的JSON字符串,下面是一個例子:
- var contact={
- name:"WangMeng",
- email:"wangmeng@some-domain.com"
- };
- var jsontext=JSON.stringify(contact);
- alert(jsontext);
這個例子中的警告框會顯示下列未經(jīng)縮進(jìn)的字符串:
(\"name\":\"wangmeng\",\"email\":\"wangmeng\":\wangment@some-domain.com\)
由于并不是所有JavaSrcipt值都可以使用JSON表示,因此結(jié)果中只會包含那些正式得到支持的值。例如,函數(shù)和undefined值無法通過JSON表示,包含它們的任何鍵默認(rèn)都將被移除。要改變這個默認(rèn)的行為,可以在第二個參數(shù)的位置傳入一個函數(shù)。在序列化過程 中每當(dāng)遇到一個不支持的數(shù)據(jù)類型時,該函數(shù)就會在被序列化的對象中作用域中運用,其參數(shù)是相應(yīng)的鍵和值。
對于JSON支持的數(shù)據(jù)類型,序列化過程中不會調(diào)用這個函數(shù),這些類型包括:字符串、數(shù)值、布爾值、null、對象、數(shù)組和Date。來看下面的例子:
- var jsontext=JSON.stringify([new Function()],function(key,value){
- if(value instanceof Function){
- return "(function)";
- }else{
- return value;
- }
- });
- alert(jsontext); //"(function)"
這個例子試圖序列化一個包含函數(shù)的數(shù)組。當(dāng)遇到函數(shù)值時,第二個參數(shù)(即過濾函數(shù))會將它轉(zhuǎn)換為字符串"(function)",該字符串將出現(xiàn)在最終結(jié)果中。
使用POST請求并將JSON文本傳遞給send()方法,可以將JSON數(shù)據(jù)發(fā)送給服務(wù)器。來看下面的例子:
- var xhr=createXHR();
- var contact={
- name:"wangmeng",
- email:"wangmeng@some-domain.com"
- };
- xhr.onreadystatechange=function(){
- if(xhr.readyState == 4){
- if((xhr.status <= 200 && xhr.status < 300) || xhr.status == 304){
- alert(xhr.responseText);
- }
- }
- };
- xhr.open("post","addcontact.php",true);
- xhr.send(JSON.stringify(contact));
這個例子是要將新聯(lián)系人信息保存到服務(wù)器,因此要將數(shù)據(jù)發(fā)送給addcontact.php文件。在根據(jù)新聯(lián)系人信息構(gòu)建好contact對象后,又將它序列化為JSON數(shù)據(jù)并傳遞給send()方法。服務(wù)器上的PHP頁面負(fù)責(zé)將接受到的JSON數(shù)據(jù)解析回原來的格式,以便服務(wù)器端代碼能夠理解,同時還會向瀏覽器發(fā)送響應(yīng)。
本文地址:http://www.yiiyaa.net/1378
【編輯推薦】