轉:http://www.cnblogs.com/Garden-blog/archive/2011/03/11/1981778.htmljavascript
Ajax 完整教程
第 1 頁 Ajax 簡介
Ajax 由 HTML、JavaScript™ 技術、DHTML 和 DOM 組成,這一傑出的方法能夠將笨拙的 Web 界面轉化成交互性的 Ajax 應用程序。本文的做者是一位 Ajax 專家,他演示了這些技術如何協同工做 —— 從整體概述到細節的討論 —— 使高效的 Web 開發成爲現實。他還揭開了 Ajax 核心概念的神祕面紗,包括 XMLHttpRequest 對象。
五年前,若是不知道 XML,您就是一隻無人重視的醜小鴨。十八個月前,Ruby 成了關注的中心,不知道 Ruby 的程序員只能坐冷板凳了。今天,若是想跟上最新的技術時尚,那您的目標就是 Ajax。
可是,Ajax 不只僅 是一種時尚,它是一種構建網站的強大方法,並且不像學習一種全新的語言那樣困難。
但在詳細探討 Ajax 是什麼以前,先讓咱們花幾分鐘瞭解 Ajax 作 什麼。目前,編寫應用程序時有兩種基本的選擇:
·桌面應用程序
·Web 應用程序
二者是相似的,桌面應用程序一般以 CD 爲介質(有時候可從網站下載)並徹底安裝到您的計算機上。桌面應用程序可能使用互聯網下載更新,但運行這些應用程序的代碼在桌面計算機上。Web 應用程序運行在某處的 Web 服務器上 —— 絕不奇怪,要經過 Web 瀏覽器訪問這種應用程序。
不 過,比這些應用程序的運行代碼放在何處更重要的是,應用程序如何運轉以及如何與其進行交互。桌面應用程序通常很快(就在您的計算機上運行,不用等待互聯網 鏈接),具備漂亮的用戶界面(一般和操做系統有關)和非凡的動態性。能夠單擊、選擇、輸入、打開菜單和子菜單、處處巡遊,基本上不須要等待。
另外一方面,Web 應用程序是最新的潮流,它們提供了在桌面上不能實現的服務(好比 Amazon.com 和 eBay)。可是,伴隨着 Web 的強大而出現的是等待,等待服務器響應,等待屏幕刷新,等待請求返回和生成新的頁面。
顯然這樣說過於簡略了,但基本的概念就是如此。您可能已經猜到,Ajax 嘗試創建桌面應用程序的功能和交互性,與不斷更新的 Web 應用程序之間的橋樑。能夠使用像桌面應用程序中常見的動態用戶界面和漂亮的控件,不過是在 Web 應用程序中。
還等什麼呢?咱們來看看 Ajax 如何將笨拙的 Web 界面轉化成能迅速響應的 Ajax 應用程序吧。
老技術,新技巧
在 談到 Ajax 時,實際上涉及到多種技術,要靈活地運用它必須深刻了解這些不一樣的技術(本系列的頭幾篇文章將分別討論這些技術)。好消息是您可能已經很是熟悉其中的大部 分技術,更好的是這些技術都很容易學習,並不像完整的編程語言(如 Java 或 Ruby)那樣困難。
下面是 Ajax 應用程序所用到的基本技術:
·HTML 用於創建 Web 表單並肯定應用程序其餘部分使用的字段。
·JavaScript 代碼是運行 Ajax 應用程序的核心代碼,幫助改進與服務器應用程序的通訊。
·DHTML 或 Dynamic HTML,用於動態更新表單。咱們將使用 div、span 和其餘動態 HTML 元素來標記 HTML。
·文檔對象模型 DOM 用於(經過 JavaScript 代碼)處理 HTML 結構和(某些狀況下)服務器返回的 XML。
Ajax 的定義
順便說一下,Ajax 是 Asynchronous JavaScript and XML(以及 DHTML 等)的縮寫。這個短語是 Adaptive Path 的 Jesse James Garrett 發明的(請參閱 參考資料),按照 Jesse 的解釋,這不是 個首字母縮寫詞。
咱們來進一步分析這些技術的職責。之後的文章中我將深刻討論這些技術,目前只要熟悉這些組件和技術就能夠了。對這些代碼越熟悉,就越容易從對這些技術的零散瞭解轉變到真正把握這些技術(同時也真正打開了 Web 應用程序開發的大門)。
XMLHttpRequest 對象
要了解的一個對象可能對您來講也是最陌生的,即 XMLHttpRequest。這是一個 JavaScript 對象,建立該對象很簡單,如清單 1 所示。
清單 1. 建立新的 XMLHttpRequest 對象
<script language="javascript" type="text/javascript">
var xmlHttp = new XMLHttpRequest();
</script>
下一期文章中將進一步討論這個對象,如今要知道這是處理全部服務器通訊的對象。繼續閱讀以前,先停下來想想:經過 XMLHttpRequest 對象與服務器進行對話的是 JavaScript 技術。這不是通常的應用程序流,這偏偏是 Ajax 的強大功能的來源。
在通常的 Web 應用程序中,用戶填寫表單字段並單擊 Submit 按鈕。而後整個表單發送到服務器,服務器將它轉發給處理表單的腳本(一般是 PHP 或 Java,也多是 CGI 進程或者相似的東西),腳本執行完成後再發送回全新的頁面。該頁面多是帶有已經填充某些數據的新表單的 HTML,也多是確認頁面,或者是具備根據原來表單中輸入數據選擇的某些選項的頁面。固然,在服務器上的腳本或程序處理和返回新表單時用戶必須等待。屏 幕變成一片空白,等到服務器返回數據後再從新繪製。這就是交互性差的緣由,用戶得不到當即反饋,所以感受不一樣於桌面應用程序。
Ajax 基本上就是把 JavaScript 技術和 XMLHttpRequest 對象放在 Web 表單和服務器之間。當用戶填寫表單時,數據發送給一些 JavaScript 代碼而不是 直接發送給服務器。相反,JavaScript 代碼捕獲表單數據並向服務器發送請求。同時用戶屏幕上的表單也不會閃爍、消失或延遲。換句話說,JavaScript 代碼在幕後發送請求,用戶甚至不知道請求的發出。更好的是,請求是異步發送的,就是說 JavaScript 代碼(和用戶)不用等待服務器的響應。所以用戶能夠繼續輸入數據、滾動屏幕和使用應用程序。
而後,服務器將數據返回 JavaScript 代碼(仍然在 Web 表單中),後者決定如何處理這些數據。它能夠迅速更新表單數據,讓人感受應用程序是當即完成的,表單沒有提交或刷新而用戶獲得了新數據。 JavaScript 代碼甚至能夠對收到的數據執行某種計算,再發送另外一個請求,徹底不須要用戶干預!這就是 XMLHttpRequest 的強大之處。它能夠根據須要自行與服務器進行交互,用戶甚至能夠徹底不知道幕後發生的一切。結果就是相似於桌面應用程序的動態、快速響應、高交互性的體 驗,可是背後又擁有互聯網的所有強大力量。
加入一些 JavaScript
獲得 XMLHttpRequest 的句柄後,其餘的 JavaScript 代碼就很是簡單了。事實上,咱們將使用 JavaScript 代碼完成很是基本的任務:
·獲取表單數據:JavaScript 代碼很容易從 HTML 表單中抽取數據併發送到服務器。
·修改表單上的數據:更新表單也很簡單,從設置字段值到迅速替換圖像。
·解析 HTML 和 XML:使用 JavaScript 代碼操縱 DOM(請參閱 下一節),處理 HTML 表單服務器返回的 XML 數據的結構。
對於前兩點,須要很是熟悉 getElementById() 方法,如 清單 2 所示。
清單 2. 用 JavaScript 代碼捕獲和設置字段值
// Get the value of the "phone" field and stuff it in a variable called phone
var phone = document.getElementById("phone").value;
// Set some values on a form using an array called response
document.getElementById("order").value = response[0];
document.getElementById("address").value = response[1];
這 裏沒有特別須要注意的地方,真是好極了!您應該認識到這裏並無很是複雜的東西。只要掌握了 XMLHttpRequest,Ajax 應用程序的其餘部分就是如 清單 2 所示的簡單 JavaScript 代碼了,混合有少許的 HTML。同時,還要用一點兒 DOM,咱們就來看看吧。
以 DOM 結束
最後還有 DOM,即文檔對象模型。可能對有些讀者來講 DOM 有點兒使人生畏,HTML 設計者不多使用它,即便 JavaScript 程序員也不大用到它,除非要完成某項高端編程任務。大量使用 DOM 的是 複雜的 Java 和 C/C++ 程序,這可能就是 DOM 被認爲難以學習的緣由。
幸運的是,在 JavaScript 技術中使用 DOM 很容易,也很是直觀。如今,按照常規也許應該說明如何使用 DOM,或者至少要給出一些示例代碼,但這樣作也可能誤導您。即便不理會 DOM,仍然能深刻地探討 Ajax,這也是我準備採用的方法。之後的文章將再次討論 DOM,如今只要知道可能須要 DOM 就能夠了。當須要在 JavaScript 代碼和服務器之間傳遞 XML 和改變 HTML 表單的時候,咱們再深刻研究 DOM。沒有它也能作一些有趣的工做,所以如今就把 DOM 放到一邊吧。
獲取 Request 對象
有了上面的基礎知識後,咱們來看看一些具體的例子。XMLHttpRequest 是 Ajax 應用程序的核心,並且對不少讀者來講可能還比較陌生,咱們就從這裏開始吧。從 清單 1 能夠看出,建立和使用這個對象很是簡單,不是嗎?等一等。
還 記得幾年前的那些討厭的瀏覽器戰爭嗎?沒有同樣東西在不一樣的瀏覽器上獲得一樣的結果。無論您是否相信,這些戰爭仍然在繼續,雖然規模較小。但使人奇怪的 是,XMLHttpRequest 成了這場戰爭的犧牲品之一。所以得到 XMLHttpRequest 對象可能須要採用不一樣的方法。下面我將詳細地進行解釋。
使用 Microsoft 瀏覽器
Microsoft 瀏覽器 Internet Explorer 使用 MSXML 解析器處理 XML(能夠經過 參考資料 進一步瞭解 MSXML)。所以若是編寫的 Ajax 應用程序要和 Internet Explorer 打交道,那麼必須用一種特殊的方式建立對象。
但並非這麼簡單。根據 Internet Explorer 中安裝的 JavaScript 技術版本不一樣,MSXML 實際上有兩種不一樣的版本,所以必須對這兩種狀況分別編寫代碼。請參閱 清單 3,其中的代碼在 Microsoft 瀏覽器上建立了一個 XMLHttpRequest。
清單 3. 在 Microsoft 瀏覽器上建立 XMLHttpRequest 對象
var xmlHttp = false;
try {
xmlHttp = new ActiveXObject("Msxml2.XMLHTTP");
} catch (e) {
try {
xmlHttp = new ActiveXObject("Microsoft.XMLHTTP");
} catch (e2) {
xmlHttp = false;
}
}
您對這些代碼可能還不徹底理解,但沒有關係。當本系列文章結束的時候,您將對 JavaScript 編程、錯誤處理、條件編譯等有更深的瞭解。如今只要緊緊記住其中的兩行代碼:
xmlHttp = new ActiveXObject("Msxml2.XMLHTTP");
和
xmlHttp = new ActiveXObject("Microsoft.XMLHTTP");。
這 兩行代碼基本上就是嘗試使用一個版本的 MSXML 建立對象,若是失敗則使用另外一個版本建立該對象。不錯吧?若是都不成功,則將 xmlHttp 變量設爲 false,告訴您的代碼出現了問題。若是出現這種狀況,多是由於安裝了非 Microsoft 瀏覽器,須要使用不一樣的代碼。
處理 Mozilla 和非 Microsoft 瀏覽器
若是選擇的瀏覽器不是 Internet Explorer,或者爲非 Microsoft 瀏覽器編寫代碼,就須要使用不一樣的代碼。事實上就是 清單 1 所示的一行簡單代碼:
var xmlHttp = new XMLHttpRequest object;。
這行簡單得多的代碼在 Mozilla、Firefox、Safari、Opera 以及基本上全部以任何形式或方式支持 Ajax 的非 Microsoft 瀏覽器中,建立了 XMLHttpRequest 對象。
結合起來
關 鍵是要支持全部 瀏覽器。誰願意編寫一個只能用於 Internet Explorer 或者非 Microsoft 瀏覽器的應用程序呢?或者更糟,要編寫一個應用程序兩次?固然不!所以代碼要同時支持 Internet Explorer 和非 Microsoft 瀏覽器。清單 4 顯示了這樣的代碼。
清單 4. 以支持多種瀏覽器的方式建立 XMLHttpRequest 對象
/* Create a new XMLHttpRequest object to talk to the Web server */
var xmlHttp = false;
/*@cc_on @*/
/*@if (@_jscript_version >= 5)
try {
xmlHttp = new ActiveXObject("Msxml2.XMLHTTP");
} catch (e) {
try {
xmlHttp = new ActiveXObject("Microsoft.XMLHTTP");
} catch (e2) {
xmlHttp = false;
}
}
@end @*/
if (!xmlHttp && typeof XMLHttpRequest != 'undefined') {
xmlHttp = new XMLHttpRequest();
}
如今先無論那些註釋掉的奇怪符號,如 @cc_on,這是特殊的 JavaScript 編譯器命令,將在下一期針對 XMLHttpRequest 的文章中詳細討論。這段代碼的核心分爲三步:
一、創建一個變量 xmlHttp 來引用即將建立的 XMLHttpRequest 對象。
二、嘗試在 Microsoft 瀏覽器中建立該對象:
1)嘗試使用 Msxml2.XMLHTTP 對象建立它。
2)若是失敗,再嘗試 Microsoft.XMLHTTP 對象。
二、若是仍然沒有創建 xmlHttp,則以非 Microsoft 的方式建立該對象。
最後,xmlHttp 應該引用一個有效的 XMLHttpRequest 對象,不管運行什麼樣的瀏覽器。
關於安全性的一點說明
安 全性如何呢?如今瀏覽器容許用戶提升他們的安全等級,關閉 JavaScript 技術,禁用瀏覽器中的任何選項。在這種狀況下,代碼不管如何都不會工做。此時必須適當地處理問題,這須要單獨的一篇文章來討論,要放到之後了(這個系列夠 長了吧?不用擔憂,讀完以前也許您就掌握了)。如今要編寫一段健壯但不夠完美的代碼,對於掌握 Ajax 來講就很好了。之後咱們還將討論更多的細節。
Ajax 世界中的請求/響應
如今咱們介紹了 Ajax,對 XMLHttpRequest 對象以及如何建立它也有了基本的瞭解。若是閱讀得很仔細,您可能已經知道與服務器上的 Web 應用程序打交道的是 JavaScript 技術,而不是直接提交給那個應用程序的 HTML 表單。
還缺乏什麼呢?到底如何使用 XMLHttpRequest。由於這段代碼很是重要,您編寫的每一個 Ajax 應用程序都要以某種形式使用它,先看看 Ajax 的基本請求/響應模型是什麼樣吧。
發出請求
您已經有了一個嶄新的 XMLHttpRequest 對象,如今讓它乾點活兒吧。首先須要一個 Web 頁面可以調用的 JavaScript 方法(好比當用戶輸入文本或者從菜單中選擇一項時)。接下來就是在全部 Ajax 應用程序中基本都雷同的流程:
一、從 Web 表單中獲取須要的數據。
二、創建要鏈接的 URL。
三、打開到服務器的鏈接。
四、設置服務器在完成後要運行的函數。
五、發送請求。
清單 5 中的示例 Ajax 方法就是按照這個順序組織的:
清單 5. 發出 Ajax 請求
function callServer() {
// Get the city and state from the web form
var city = document.getElementById("city").value;
var state = document.getElementById("state").value;
// Only go on if there are values for both fields
if ((city == null) || (city == "")) return;
if ((state == null) || (state == "")) return;
// Build the URL to connect to
var url = "/scripts/getZipCode.php?city=" + escape(city) + "&state=" + escape(state);
// Open a connection to the server
xmlHttp.open("GET", url, true);
// Setup a function for the server to run when it's done
xmlHttp.onreadystatechange = updatePage;
// Send the request
xmlHttp.send(null);
}
其中大部分代碼意義都很明確。開始的代碼使用基本 JavaScript 代碼獲取幾個表單字段的值。而後設置一個 PHP 腳本做爲連接的目標。要注意腳本 URL 的指定方式,city 和 state(來自表單)使用簡單的 GET 參數附加在 URL 以後。
然 後打開一個鏈接,這是您第一次看到使用 XMLHttpRequest。其中指定了鏈接方法(GET)和要鏈接的 URL。最後一個參數若是設爲 true,那麼將請求一個異步鏈接(這就是 Ajax 的由來)。若是使用 false,那麼代碼發出請求後將等待服務器返回的響應。若是設爲 true,當服務器在後臺處理請求的時候用戶仍然能夠使用表單(甚至調用其餘 JavaScript 方法)。
xmlHttp(要記住, 這是 XMLHttpRequest 對象實例)的 onreadystatechange 屬性能夠告訴服務器在運行完成 後(可能要用五分鐘或者五個小時)作什麼。由於代碼沒有等待服務器,必須讓服務器知道怎麼作以便您能做出響應。在這個示例中,若是服務器處理完了請求,一 個特殊的名爲 updatePage() 的方法將被觸發。
最後,使用值 null 調用 send()。由於已經在請求 URL 中添加了要發送給服務器的數據(city 和 state),因此請求中不須要發送任何數據。這樣就發出了請求,服務器按照您的要求工做。
若是沒有發現任何新鮮的東西,您應該體會到這是多麼簡單明瞭!除了緊緊記住 Ajax 的異步特性外,這些內容都至關簡單。應該感激 Ajax 使您可以專心編寫漂亮的應用程序和界面,而不用擔憂複雜的 HTTP 請求/響應代碼。
清單 5 中的代碼說明了 Ajax 的易用性。數據是簡單的文本,能夠做爲請求 URL 的一部分。用 GET 而不是更復雜的 POST 發送請求。沒有 XML 和要添加的內容頭部,請求體中沒有要發送的數據;換句話說,這就是 Ajax 的烏托邦。
不 用擔憂,隨着本系列文章的展開,事情會變得愈來愈複雜。您將看到如何發送 POST 請求、如何設置請求頭部和內容類型、如何在消息中編碼 XML、如何增長請求的安全性,能夠作的工做還有不少!暫時先不用管那些難點,掌握好基本的東西就好了,很快咱們就會創建一整套的 Ajax 工具庫。
處理響應
如今要面對服務器的響應了。如今只要知道兩點:
·什麼也不要作,直到 xmlHttp.readyState 屬性的值等於 4。
·服務器將把響應填充到 xmlHttp.responseText 屬性中。
其 中的第一點,即就緒狀態,將在下一篇文章中詳細討論,您將進一步瞭解 HTTP 請求的階段,可能比您設想的還多。如今只要檢查一個特定的值(4)就能夠了(下一期文章中還有更多的值要介紹)。第二點,使用 xmlHttp.responseText 屬性得到服務器的響應,這很簡單。清單 6 中的示例方法可供服務器根據 清單 5 中發送的數據調用。
清單 6. 處理服務器響應
function updatePage() {
if (xmlHttp.readyState == 4) {
var response = xmlHttp.responseText;
document.getElementById("zipCode").value = response;
}
}
這 些代碼一樣既不難也不復雜。它等待服務器調用,若是是就緒狀態,則使用服務器返回的值(這裏是用戶輸入的城市和州的 ZIP 編碼)設置另外一個表單字段的值。因而包含 ZIP 編碼的 zipCode 字段忽然出現了,而用戶沒有按任何按鈕!這就是前面所說的桌面應用程序的感受。快速響應、動態感覺等等,這些都只由於有了小小的一段 Ajax 代碼。
細 心的讀者可能注意到 zipCode 是一個普通的文本字段。一旦服務器返回 ZIP 編碼,updatePage() 方法就用城市/州的 ZIP 編碼設置那個字段的值,用戶就能夠改寫該值。這樣作有兩個緣由:保持例子簡單,說明有時候可能但願 用戶可以修改服務器返回的數據。要記住這兩點,它們對於好的用戶界面設計來講很重要。php
鏈接 Web 表單
還有什麼呢?實際上 沒有多少了。一個 JavaScript 方法捕捉用戶輸入表單的信息並將其發送到服務器,另外一個 JavaScript 方法監聽和處理響應,並在響應返回時設置字段的值。全部這些實際上都依賴於調用 第一個 JavaScript 方法,它啓動了整個過程。最明顯的辦法是在 HTML 表單中增長一個按鈕,但這是 2001 年的辦法,您不這樣認爲嗎?仍是像 清單 7 這樣利用 JavaScript 技術吧。
清單 7. 啓動一個 Ajax 過程
<form>
<p>City: <input type="text" name="city" id="city" size="25"
onChange="callServer();" /></p>
<p>State: <input type="text" name="state" id="state" size="25"
onChange="callServer();" /></p>
<p>Zip Code: <input type="text" name="zipCode" id="city" size="5" /></p>
</form>
若是感受這像是一段至關普通的代碼,那就對了,正是如此!當用戶在 city 或 state 字段中輸入新的值時,callServer() 方法就被觸發,因而 Ajax 開始運行了。有點兒明白怎麼回事了吧?好,就是如此!
結束語
現 在您可能已經準備開始編寫第一個 Ajax 應用程序了,至少也但願認真讀一下 參考資料 中的那些文章了吧?但能夠首先從這些應用程序如何工做的基本概念開始,對 XMLHttpRequest 對象有基本的瞭解。在下一期文章中,您將掌握這個對象,學會如何處理 JavaScript 和服務器的通訊、如何使用 HTML 表單以及如何得到 DOM 句柄。
如今先花點兒時間考慮考慮 Ajax 應用程序有多麼強大。設想一下,當單擊按鈕、輸入一個字段、從組合框中選擇一個選項或者用鼠標在屏幕上拖動時,Web 表單可以馬上做出響應會是什麼情形。想想異步 究竟意味着什麼,想想 JavaScript 代碼運行並且不等待 服務器對它的請求做出響應。會遇到什麼樣的問題?會進入什麼樣的領域?考慮到這種新的方法,編程的時候應如何改變表單的設計?
若是在這些問題上花一點兒時間,與簡單地剪切/粘貼某些代碼到您根本不理解的應用程序中相比,收益會更多。在下一期文章中,咱們將把這些概念付諸實踐,詳細介紹使應用程序按照這種方式工做所須要的代碼。所以,如今先享受一下 Ajax 所帶來的可能性吧。 html
第 2 頁 使用 JavaScript 和 Ajax 發出異步請求
多 數 Web 應用程序都使用請求/響應模型從服務器上得到完整的 HTML 頁面。經常是點擊一個按鈕,等待服務器響應,再點擊另外一個按鈕,而後再等待,這樣一個反覆的過程。有了 Ajax 和 XMLHttpRequest 對象,就能夠使用沒必要讓用戶等待服務器響應的請求/響應模型了。本文中,Brett McLaughlin 介紹瞭如何建立可以適應不一樣瀏覽器的 XMLHttpRequest 實例,創建和發送請求,並響應服務器。
本系列的上一期文章(請參閱 參考資料 中的連接),咱們介紹了 Ajax 應用程序,考察了推進 Ajax 應用程序的基本概念。其中的核心是不少您可能已經瞭解的技術:JavaScript、HTML 和 XHTML、一點動態 HTML 以及 DOM(文檔對象模型)。本文將放大其中的一點,把目光放到具體的 Ajax 細節上。
本文中, 您將開始接觸最基本和基礎性的有關 Ajax 的所有對象和編程方法:XMLHttpRequest 對象。該對象實際上僅僅是一個跨越全部 Ajax 應用程序的公共線程,您可能已經預料到,只有完全理解該對象才能充分發揮編程的潛力。事實上,有時您會發現,要正確地使用 XMLHttpRequest,顯然不能 使用 XMLHttpRequest。這究竟是怎麼回事呢?
Web 2.0 一瞥
在 深刻研究代碼以前首先看看最近的觀點 —— 必定要十分清楚 Web 2.0 這個概念。聽到 Web 2.0 這個詞的時候,應該首先問一問 「Web 1.0 是什麼?」 雖然不多聽人提到 Web 1.0,實際上它指的就是具備徹底不一樣的請求和響應模型的傳統 Web。好比,到 Amazon.com 網站上點擊一個按鈕或者輸入搜索項。就會對服務器發送一個請求,而後響應再返回到瀏覽器。該請求不只僅是圖書和書目列表,而是另外一個完整的 HTML 頁面。所以當 Web 瀏覽器用新的 HTML 頁面重繪時,可能會看到閃爍或抖動。事實上,經過看到的每一個新頁面能夠清晰地看到請求和響應。
Web 2.0(在很大程度上)消除了這種看得見的往復交互。好比訪問 Google Maps 或 Flickr 這樣的站點(到這些支持 Web 2.0 和 Ajax 站點的連接請參閱 參考資料)。好比在 Google Maps 上,您能夠拖動地圖,放大和縮小,只有不多的重繪操做。固然這裏仍然有請求和響應,只不過都藏到了幕後。做爲用戶,體驗更加溫馨,感受很像桌面應用程序。 這種新的感覺和範型就是當有人提到 Web 2.0 時您所體會到的。
須要關心的是如何使這些新的交互成爲可能。顯然,仍然須要發出請求 和接收響應,但正是針對每次請求/響應交互的 HTML 重繪形成了緩慢、笨拙的 Web 交互的感覺。所以很清楚,咱們須要一種方法使發送的請求和接收的響應只 包含須要的數據而不是整個 HTML 頁面。唯一須要得到整個新 HTML 頁面的時候就是但願用戶看到 新頁面的時候。
但多數交互都是在已有頁面上增長細節、修改主體文本或者覆蓋原有數據。這些狀況下,Ajax 和 Web 2.0 方法容許在不 更新整個 HTML 頁面的狀況下發送和接收數據。對於那些常常上網的人,這種能力可讓您的應用程序感受更快、響應更及時,讓他們不時地光顧您的網站。
XMLHttpRequest 簡介
要 真正實現這種絢麗的奇蹟,必須很是熟悉一個 JavaScript 對象,即 XMLHttpRequest。這個小小的對象實際上已經在幾種瀏覽器中存在一段時間了,它是本專欄從此幾個月中要介紹的 Web 2.0、Ajax 和大部分其餘內容的核心。爲了讓您快速地大致瞭解它,下面給出將要用於該對象的不多的幾個 方法和屬性。
·open():創建到服務器的新請求。
·send():向服務器發送請求。
·abort():退出當前請求。
·readyState:提供當前 HTML 的就緒狀態。
·responseText:服務器返回的請求響應文本。
如 果不瞭解這些(或者其中的任何 一個),您也不用擔憂,後面幾篇文章中咱們將介紹每一個方法和屬性。如今應該 瞭解的是,明確用 XMLHttpRequest 作什麼。要注意這些方法和屬性都與發送請求及處理響應有關。事實上,若是看到 XMLHttpRequest 的全部方法和屬性,就會發現它們都 與很是簡單的請求/響應模型有關。顯然,咱們不會遇到特別新的 GUI 對象或者建立用戶交互的某種超極神祕的方法,咱們將使用很是簡單的請求和很是簡單的響應。聽起來彷佛沒有多少吸引力,可是用好該對象能夠完全改變您的應用 程序。
簡單的 new
首先須要建立一個新變量並賦給它一個 XMLHttpRequest 對象實例。這在 JavaScript 中很簡單,只要對該對象名使用 new 關鍵字便可,如 清單 1 所示。
清單 1. 建立新的 XMLHttpRequest 對象
<script language="javascript" type="text/javascript">
var request = new XMLHttpRequest();
</script>
不難吧?記住,JavaScript 不要求指定變量類型,所以不須要像 清單 2 那樣作(在 Java 語言中可能須要這樣)。
清單 2. 建立 XMLHttpRequest 的 Java 僞代碼
XMLHttpRequest request = new XMLHttpRequest();
所以在 JavaScript 中用 var 建立一個變量,給它一個名字(如 「request」),而後賦給它一個新的 XMLHttpRequest 實例。此後就能夠在函數中使用該對象了。
錯誤處理
在 實際上各類事情均可能出錯,而上面的代碼沒有提供任何錯誤處理。較好的辦法是建立該對象,並在出現問題時優雅地退出。好比,任何較早的瀏覽器(不論您是否 相信,仍然有人在使用老版本的 Netscape Navigator)都不支持 XMLHttpRequest,您須要讓這些用戶知道有些地方出了問題。清單 3 說明如何建立該對象,以便在出現問題的時候發出 JavaScript 警告。
清單 3. 建立具備錯誤處理能力的 XMLHttpRequest
<script language="javascript" type="text/javascript">
var request = false;
try {
request = new XMLHttpRequest();
} catch (failed) {
request = false;
}
if (!request)
alert("Error initializing XMLHttpRequest!");
</script>
必定要理解這些步驟:
一、建立一個新變量 request 並賦值 false。後面將使用 false 做爲斷定條件,它表示尚未建立 XMLHttpRequest 對象。
二、增長 try/catch 塊:
1)嘗試建立 XMLHttpRequest 對象。
2)若是失敗(catch (failed))則保證 request 的值仍然爲 false。
三、檢查 request 是否仍爲 false(若是一切正常就不會是 false)。
四、若是出現問題(request 是 false)則使用 JavaScript 警告通知用戶出現了問題。
代碼很是簡單,對大多數 JavaScript 和 Web 開發人員來講,真正理解它要比讀寫代碼花更長的時間。如今已經獲得了一段帶有錯誤檢查的 XMLHttpRequest 對象建立代碼,還能夠告訴您哪兒出了問題。
應付 Microsoft
看起來彷佛一切良好,至少在用 Internet Explorer 試驗這些代碼以前是這樣的。若是這樣試驗的話,就會看到 圖 1 所示的糟糕情形。
圖 1. Internet Explorer 報告錯誤
顯 然有什麼地方不對勁,而 Internet Explorer 很難說是一種過期的瀏覽器,由於全世界有 70% 在使用 Internet Explorer。換句話說,若是不支持 Microsoft 和 Internet Explorer 就不會受到 Web 世界的歡迎!所以咱們須要採用不一樣的方法處理 Microsoft 瀏覽器。
經驗證發現 Microsoft 支持 Ajax,可是其 XMLHttpRequest 版本有不一樣的稱呼。事實上,它將其稱爲幾種 不一樣的東西。若是使用較新版本的 Internet Explorer,則須要使用對象 Msxml2.XMLHTTP,而較老版本的 Internet Explorer 則使用 Microsoft.XMLHTTP。咱們須要支持這兩種對象類型(同時還要支持非 Microsoft 瀏覽器)。請看看 清單 4,它在前述代碼的基礎上增長了對 Microsoft 的支持。
Microsoft 參與了嗎?
關於 Ajax 和 Microsoft 對該領域不斷增加的興趣和參與已經有不少文章進行了介紹。事實上,聽說 Microsoft 最新版本的 Internet Explorer —— version 7.0,將在 2006 年下半年推出 —— 將開始直接支持 XMLHttpRequest,讓您使用 new 關鍵字代替全部的 Msxml2.XMLHTTP 建立代碼。但不要太激動,仍然須要支持舊的瀏覽器,所以跨瀏覽器代碼不會很快消失。
清單 4. 增長對 Microsoft 瀏覽器的支持
<script language="javascript" type="text/javascript">
var request = false;
try {
request = new XMLHttpRequest();
} catch (trymicrosoft) {
try {
request = new ActiveXObject("Msxml2.XMLHTTP");
} catch (othermicrosoft) {
try {
request = new ActiveXObject("Microsoft.XMLHTTP");
} catch (failed) {
request = false;
}
}
}
if (!request)
alert("Error initializing XMLHttpRequest!");
</script>
很容易被這些花括號迷住了眼睛,所以下面分別介紹每一步:
一、建立一個新變量 request 並賦值 false。使用 false 做爲判斷條件,它表示尚未建立 XMLHttpRequest 對象。
二、增長 try/catch 塊:
1)嘗試建立 XMLHttpRequest 對象。
2)若是失敗(catch (trymicrosoft)):
1>嘗試使用較新版本的 Microsoft 瀏覽器建立 Microsoft 兼容的對象(Msxml2.XMLHTTP)。
2> 若是失敗(catch (othermicrosoft))嘗試使用較老版本的 Microsoft 瀏覽器建立 Microsoft 兼容的對象(Microsoft.XMLHTTP)。
2)若是失敗(catch (failed))則保證 request 的值仍然爲 false。
三、檢查 request 是否仍然爲 false(若是一切順利就不會是 false)。
四、若是出現問題(request 是 false)則使用 JavaScript 警告通知用戶出現了問題。
這樣修改代碼以後再使用 Internet Explorer 試驗,就應該看到已經建立的表單(沒有錯誤消息)。我實驗的結果如 圖 2 所示。
圖 2. Internet Explorer 正常工做
靜態與動態
再 看一看清單 一、3 和 4,注意,全部這些代碼都直接嵌套在 script 標記中。像這種不放到方法或函數體中的 JavaScript 代碼稱爲靜態 JavaScript。就是說代碼是在頁面顯示給用戶以前的某個時候運行。(雖然根據規範不能徹底精確地 知道這些代碼什麼時候運行對瀏覽器有什麼影響,可是能夠保證這些代碼在用戶可以與頁面交互以前運行。)這也是多數 Ajax 程序員建立 XMLHttpRequest 對象的通常方式。
就是說,也能夠像 清單 5 那樣將這些代碼放在一個方法中。
清單 5. 將 XMLHttpRequest 建立代碼移動到方法中
<script language="javascript" type="text/javascript">
var request;
function createRequest() {
try {
request = new XMLHttpRequest();
} catch (trymicrosoft) {
try {
request = new ActiveXObject("Msxml2.XMLHTTP");
} catch (othermicrosoft) {
try {
request = new ActiveXObject("Microsoft.XMLHTTP");
} catch (failed) {
request = false;
}
}
}
if (!request)
alert("Error initializing XMLHttpRequest!");
}
</script>
若是按照這種方式編寫代碼,那麼在處理 Ajax 以前須要調用該方法。所以還須要 清單 6 這樣的代碼。
清單 6. 使用 XMLHttpRequest 的建立方法
<script language="javascript" type="text/javascript">
var request;
function createRequest() {
try {
request = new XMLHttpRequest();
} catch (trymicrosoft) {
try {
request = new ActiveXObject("Msxml2.XMLHTTP");
} catch (othermicrosoft) {
try {
request = new ActiveXObject("Microsoft.XMLHTTP");
} catch (failed) {
request = false;
}
}
}
if (!request)
alert("Error initializing XMLHttpRequest!");
}
function getCustomerInfo() {
createRequest();
// Do something with the request variable
}
</script>
此 代碼唯一的問題是推遲了錯誤通知,這也是多數 Ajax 程序員不採用這一方法的緣由。假設一個複雜的表單有 10 或 15 個字段、選擇框等,當用戶在第 14 個字段(按照表單順序從上到下)輸入文本時要激活某些 Ajax 代碼。這時候運行 getCustomerInfo() 嘗試建立一個 XMLHttpRequest 對象,但(對於本例來講)失敗了。而後向用戶顯示一條警告,明確地告訴他們不能使用該應用程序。但用戶已經花費了不少時間在表單中輸入數據!這是很是使人 討厭的,而討厭顯然不會吸引用戶再次訪問您的網站。
若是使用靜態 JavaScript,用戶在點擊頁面的時候很快就會看到錯誤信息。這樣也很煩人,是否是?可能令用戶錯誤地認爲您的 Web 應用程序不能在他的瀏覽器上運行。不過,固然要比他們花費了 10 分鐘輸入信息以後再顯示一樣的錯誤要好。所以,我建議編寫靜態的代碼,讓用戶儘量早地發現問題。
用 XMLHttpRequest 發送請求
得 到請求對象以後就能夠進入請求/響應循環了。記住,XMLHttpRequest 唯一的目的是讓您發送請求和接收響應。其餘一切都是 JavaScript、CSS 或頁面中其餘代碼的工做:改變用戶界面、切換圖像、解釋服務器返回的數據。準備好 XMLHttpRequest 以後,就能夠向服務器發送請求了。
歡迎使用沙箱
Ajax 採用一種沙箱安全模型。所以,Ajax 代碼(具體來講就是 XMLHttpRequest 對象)只能對所在的同一個域發送請求。之後的文章中將進一步介紹安全和 Ajax,如今只要知道在本地機器上運行的代碼只能對本地機器上的服務器端腳本發送請求。若是讓 Ajax 代碼在
http://www.breakneckpizza.com/ 上運行,則必須
http://www.breakneck.com/ 中運行的腳本發送請求。
設置服務器 URL
首 先要肯定鏈接的服務器的 URL。這並非 Ajax 的特殊要求,但仍然是創建鏈接所必需的,顯然如今您應該知道如何構造 URL 了。多數應用程序中都會結合一些靜態數據和用戶處理的表單中的數據來構造該 URL。好比,清單 7 中的 JavaScript 代碼獲取電話號碼字段的值並用其構造 URL。
清單 7. 創建請求 URL
<script language="javascript" type="text/javascript">
var request = false;
try {
request = new XMLHttpRequest();
} catch (trymicrosoft) {
try {
request = new ActiveXObject("Msxml2.XMLHTTP");
} catch (othermicrosoft) {
try {
request = new ActiveXObject("Microsoft.XMLHTTP");
} catch (failed) {
request = false;
}
}
}
if (!request)
alert("Error initializing XMLHttpRequest!");
function getCustomerInfo() {
var phone = document.getElementById("phone").value;
var url = "/cgi-local/lookupCustomer.php?phone=" + escape(phone);
}
</script>
這裏沒有難懂的地方。首先,代碼建立了一個新變量 phone,並把 ID 爲 「phone」 的表單字段的值賦給它。清單 8 展現了這個表單的 XHTML,其中能夠看到 phone 字段及其 id 屬性。
清單 8. Break Neck Pizza 表單
<body>
<p><img src="breakneck-logo_4c.gif" alt="Break Neck Pizza" /></p>
<form action="POST">
<p>Enter your phone number:
<input type="text" size="14" name="phone" id="phone"
onChange="getCustomerInfo();" />
</p>
<p>Your order will be delivered to:</p>
<div id="address"></div>
<p>Type your order in here:</p>
<p><textarea name="order" rows="6" cols="50" id="order"></textarea></p>
<p><input type="submit" value="Order Pizza" id="submit" /></p>
</form>
</body>
還 要注意,當用戶輸入電話號碼或者改變電話號碼時,將觸發 清單 8 所示的 getCustomerInfo() 方法。該方法取得電話號碼並構造存儲在 url 變量中的 URL 字符串。記住,因爲 Ajax 代碼是沙箱型的,於是只能鏈接到同一個域,實際上 URL 中不須要域名。該例中的腳本名爲 /cgi-local/lookupCustomer.php。最後,電話號碼做爲 GET 參數附加到該腳本中:"phone=" + escape(phone)。
若是之前沒用見過 escape() 方法,它用於轉義不能用明文正確發送的任何字符。好比,電話號碼中的空格將被轉換成字符 %20,從而可以在 URL 中傳遞這些字符。
能夠根據須要添加任意多個參數。好比,若是須要增長另外一個參數,只須要將其附加到 URL 中並用 「與」(&)字符分開 [第一個參數用問號(?)和腳本名分開]。
打開請求
有了要鏈接的 URL 後就能夠配置請求了。能夠用 XMLHttpRequest 對象的 open() 方法來完成。該方法有五個參數:
request-type:發送請求的類型。典型的值是 GET 或 POST,但也能夠發送 HEAD 請求。
url:要鏈接的 URL。
asynch:若是但願使用異步鏈接則爲 true,不然爲 false。該參數是可選的,默認爲 true。
username:若是須要身份驗證,則能夠在此指定用戶名。該可選參數沒有默認值。 password:若是須要身份驗證,則能夠在此指定口令。該可選參數沒有默認值。
open() 是打開嗎?
Internet 開發人員對 open() 方法到底作什麼沒有達成一致。但它實際上並非 打開一個請求。若是監控 XHTML/Ajax 頁面及其鏈接腳本之間的網絡和數據傳遞,當調用 open() 方法時將看不到任何通訊。不清楚爲什麼選用了這個名字,但顯然不是一個好的選擇。
一般使用其中的前三個參數。事實上,即便須要異步鏈接,也應該指定第三個參數爲 「true」。這是默認值,但堅持明確指定請求是異步的仍是同步的更容易理解。
將這些結合起來,一般會獲得 清單 9 所示的一行代碼。
清單 9. 打開請求
function getCustomerInfo() {
var phone = document.getElementById("phone").value;
var url = "/cgi-local/lookupCustomer.php?phone=" + escape(phone);
request.open("GET", url, true);
}
一旦設置好了 URL,其餘就簡單了。多數請求使用 GET 就夠了(後面的文章中將看到須要使用 POST 的狀況),再加上 URL,這就是使用 open() 方法須要的所有內容了。
挑戰異步性
本 系列的後面一篇文章中,我將用不少時間編寫和使用異步代碼,可是您應該明白爲何 open() 的最後一個參數這麼重要。在通常的請求/響應模型中,好比 Web 1.0,客戶機(瀏覽器或者本地機器上運行的代碼)向服務器發出請求。該請求是同步的,換句話說,客戶機等待服務器的響應。當客戶機等待的時候,至少會用 某種形式通知您在等待:
·沙漏(特別是 Windows 上)。
·旋轉的皮球(一般在 Mac 機器上)。
·應用程序基本上凍結了,而後過一段時間光標變化了。
這正是 Web 應用程序讓人感到笨拙或緩慢的緣由 —— 缺少真正的交互性。按下按鈕時,應用程序實際上變得不能使用,直到剛剛觸發的請求獲得響應。若是請求須要大量服務器處理,那麼等待的時間可能很長(至少在這個多處理器、DSL 沒有等待的世界中是如此)。
而 異步請求不 等待服務器響應。發送請求後應用程序繼續運行。用戶仍然能夠在 Web 表單中輸入數據,甚至離開表單。沒有旋轉的皮球或者沙漏,應用程序也沒有明顯的凍結。服務器悄悄地響應請求,完成後告訴原來的請求者工做已經結束(具體的 辦法很快就會看到)。結果是,應用程序感受不 那麼遲鈍或者緩慢,而是響應迅速、交互性強,感受快多了。這僅僅是 Web 2.0 的一部分,但它是很重要的一部分。全部老套的 GUI 組件和 Web 設計範型都不能克服緩慢、同步的請求/響應模型。
發送請求
一旦用 open() 配置好以後,就能夠發送請求了。幸運的是,發送請求的方法的名稱要比 open() 適當,它就是 send()。
send() 只有一個參數,就是要發送的內容。可是在考慮這個方法以前,回想一下前面已經經過 URL 自己發送過數據了:
var url = "/cgi-local/lookupCustomer.php?phone=" + escape(phone);
雖 然能夠使用 send() 發送數據,但也能經過 URL 自己發送數據。事實上,GET 請求(在典型的 Ajax 應用中大約佔 80%)中,用 URL 發送數據要容易得多。若是須要發送安全信息或 XML,可能要考慮使用 send() 發送內容(本系列的後續文章中將討論安全數據和 XML 消息)。若是不須要經過 send() 傳遞數據,則只要傳遞 null 做爲該方法的參數便可。所以您會發如今本文中的例子中只須要這樣發送請求(參見 清單 10)。
清單 10. 發送請求
function getCustomerInfo() {
var phone = document.getElementById("phone").value;
var url = "/cgi-local/lookupCustomer.php?phone=" + escape(phone);
request.open("GET", url, true);
request.send(null);
}
指定回調方法
現 在咱們所作的只有不多一點是新的、革命性的或異步的。必須認可,open() 方法中 「true」 這個小小的關鍵字創建了異步請求。可是除此以外,這些代碼與用 Java servlet 及 JSP、PHP 或 Perl 編程沒有什麼兩樣。那麼 Ajax 和 Web 2.0 最大的祕密是什麼呢?祕密就在於 XMLHttpRequest 的一個簡單屬性 onreadystatechange。
首先必定要理解這些代碼中的流程(若是須要請回顧 清單 10)。創建其請求而後發出請求。此外,由於是異步請求,因此 JavaScript 方法(例子中的 getCustomerInfo())不會等待服務器。所以代碼將繼續執行,就是說,將退出該方法而把控制返回給表單。用戶能夠繼續輸入信息,應用程序不 會等待服務器。
這就提出了一個有趣的問題:服務器完成了請求以後會發生什麼?答案是什麼也不發生,至少對如今的代碼而言如此!顯然這樣不行,所以服務器在完成經過 XMLHttpRequest 發送給它的請求處理以後須要某種指示說明怎麼作。
在 JavaScript 中引用函數:
JavaScript 是一種弱類型的語言,能夠用變量引用任何東西。所以若是聲明瞭一個函數 updatePage(),JavaScript 也將該函數名看做是一個變量。換句話說,可用變量名 updatePage 在代碼中引用函數。
清單 11. 設置回調方法
function getCustomerInfo() {
var phone = document.getElementById("phone").value;
var url = "/cgi-local/lookupCustomer.php?phone=" + escape(phone);
request.open("GET", url, true);
request.onreadystatechange = updatePage;
request.send(null);
}
須要特別注意的是該屬性在代碼中設置的位置 —— 它是在調用 send() 以前 設置的。發送請求以前必須設置該屬性,這樣服務器在回答完成請求以後才能查看該屬性。如今剩下的就只有編寫 updatePage() 方法了,這是本文最後一節要討論的重點。
處理服務器響應
發 送請求,用戶高興地使用 Web 表單(同時服務器在處理請求),而如今服務器完成了請求處理。服務器查看 onreadystatechange 屬性肯定要調用的方法。除此之外,能夠將您的應用程序看做其餘應用程序同樣,不管是否異步。換句話說,不必定要採起特殊的動做編寫響應服務器的方法,只需 要改變表單,讓用戶訪問另外一個 URL 或者作響應服務器須要的任何事情。這一節咱們重點討論對服務器的響應和一種典型的動做 —— 即時改變用戶看到的表單中的一部分。
回調和 Ajax
如今咱們已經看到如何告訴服務器完成後應該作什麼:將 XMLHttpRequest 對象的 onreadystatechange 屬性設置爲要運行的函數名。這樣,當服務器處理完請求後就會自動調用該函數。也不須要擔憂該函數的任何參數。咱們從一個簡單的方法開始,如 清單 12 所示。
清單 12. 回調方法的代碼
<script language="javascript" type="text/javascript">
var request = false;
try {
request = new XMLHttpRequest();
} catch (trymicrosoft) {
try {
request = new ActiveXObject("Msxml2.XMLHTTP");
} catch (othermicrosoft) {
try {
request = new ActiveXObject("Microsoft.XMLHTTP");
} catch (failed) {
request = false;
}
}
}
if (!request)
alert("Error initializing XMLHttpRequest!");
function getCustomerInfo() {
var phone = document.getElementById("phone").value;
var url = "/cgi-local/lookupCustomer.php?phone=" + escape(phone);
request.open("GET", url, true);
request.onreadystatechange = updatePage;
request.send(null);
}
function updatePage() {
alert("Server is done!");
}
</script>
它 僅僅發出一些簡單的警告,告訴您服務器何時完成了任務。在本身的網頁中試驗這些代碼,而後在瀏覽器中打開(若是但願查看該例中的 XHTML,請參閱 清單 8)。輸入電話號碼而後離開該字段,將看到一個彈出的警告窗口(如 圖 3 所示),可是點擊 OK 又出現了……
圖 3. 彈出警告的 Ajax 代碼
根據瀏覽器的不一樣,在表單中止彈出警告以前會看到兩次、三次甚至四次警告。這是怎麼回事呢?原來咱們尚未考慮 HTTP 就緒狀態,這是請求/響應循環中的一個重要部分。
HTTP 就緒狀態
前 面提到,服務器在完成請求以後會在 XMLHttpRequest 的 onreadystatechange 屬性中查找要調用的方法。這是真的,但還不完整。事實上,每當 HTTP 就緒狀態改變時它都會調用該方法。這意味着什麼呢?首先必須理解 HTTP 就緒狀態。
HTTP 就緒狀態表示請求的狀態或情形。它用於肯定該請求是否已經開始、是否獲得了響應或者請求/響應模型是否已經完成。它還能夠幫助肯定讀取服務器提供的響應文本或數據是否安全。在 Ajax 應用程序中須要瞭解五種就緒狀態:
·0:請求沒有發出(在調用 open() 以前)。
·1:請求已經創建但尚未發出(調用 send() 以前)。
·2:請求已經發出正在處理之中(這裏一般能夠從響應獲得內容頭部)。
·3:請求已經處理,響應中一般有部分數據可用,可是服務器尚未完成響應。
·4:響應已完成,能夠訪問服務器響應並使用它。
與 大多數跨瀏覽器問題同樣,這些就緒狀態的使用也不盡一致。您也許指望任務就緒狀態從 0 到 一、二、3 再到 4,但實際上不多是這種狀況。一些瀏覽器從不報告 0 或 1 而直接從 2 開始,而後是 3 和 4。其餘瀏覽器則報告全部的狀態。還有一些則屢次報告就緒狀態 1。在上一節中看到,服務器屢次調用 updatePage(),每次調用都會彈出警告框 —— 可能和預期的不一樣!
對於 Ajax 編程,須要直接處理的唯一狀態就是就緒狀態 4,它表示服務器響應已經完成,能夠安全地使用響應數據了。基於此,回調方法中的第一行應該如 清單 13 所示。
清單 13. 檢查就緒狀態
function updatePage() {
if (request.readyState == 4)
alert("Server is done!");
}
修改後就能夠保證服務器的處理已經完成。嘗試運行新版本的 Ajax 代碼,如今就會看到與預期的同樣,只顯示一次警告信息了。
HTTP 狀態碼
雖 然 清單 13 中的代碼看起來彷佛不錯,可是還有一個問題 —— 若是服務器響應請求並完成了處理可是報告了一個錯誤怎麼辦?要知道,服務器端代碼應該明白它是由 Ajax、JSP、普通 HTML 表單或其餘類型的代碼調用的,但只能使用傳統的 Web 專用方法報告信息。而在 Web 世界中,HTTP 代碼能夠處理請求中可能發生的各類問題。
比 方說,您確定遇到過輸入了錯誤的 URL 請求而獲得 404 錯誤碼的情形,它表示該頁面不存在。這僅僅是 HTTP 請求可以收到的衆多錯誤碼中的一種(完整的狀態碼列表請參閱 參考資料 中的連接)。表示所訪問數據受到保護或者禁止訪問的 403 和 401 也很常見。不管哪一種狀況,這些錯誤碼都是從完成的響應 獲得的。換句話說,服務器履行了請求(即 HTTP 就緒狀態是 4)可是沒有返回客戶機預期的數據。
所以除了就緒狀態外,還須要檢查 HTTP 狀態。咱們指望的狀態碼是 200,它表示一切順利。若是就緒狀態是 4 並且狀態碼是 200,就能夠處理服務器的數據了,並且這些數據應該就是要求的數據(而不是錯誤或者其餘有問題的信息)。所以還要在回調方法中增長狀態檢查,如 清單 14 所示。
清單 14. 檢查 HTTP 狀態碼
function updatePage() {
if (request.readyState == 4)
if (request.status == 200)
alert("Server is done!");
}
爲了增長更健壯的錯誤處理並儘可能避免過於複雜,能夠增長一兩個狀態碼檢查,請看一看 清單 15 中修改後的 updatePage() 版本。
清單 15. 增長一點錯誤檢查
function updatePage() {
if (request.readyState == 4)
if (request.status == 200)
alert("Server is done!");
else if (request.status == 404)
alert("Request URL does not exist");
else
alert("Error: status code is " + request.status);
}
如今將 getCustomerInfo() 中的 URL 改成不存在的 URL 看看會發生什麼。應該會看到警告信息說明要求的 URL 不存在 —— 好極了!很難處理全部的錯誤條件,可是這一小小的改變可以涵蓋典型 Web 應用程序中 80% 的問題。
讀取響應文本
如今能夠確保請求已經處理完成(經過就緒狀態),服務器給出了正常的響應(經過狀態碼),最後咱們能夠處理服務器返回的數據了。返回的數據保存在 XMLHttpRequest 對象的 responseText 屬性中。
關於 responseText 中的文本內容,好比格式和長度,有意保持含糊。這樣服務器就能夠將文本設置成任何內容。比方說,一種腳本可能返回逗號分隔的值,另外一種則使用管道符(即 | 字符)分隔的值,還有一種則返回長文本字符串。何去何從由服務器決定。
在本文使用的例子中,服務器返回客戶的上一個訂單和客戶地址,中間用管道符分開。而後使用訂單和地址設置表單中的元素值,清單 16 給出了更新顯示內容的代碼。
清單 16. 處理服務器響應
function updatePage() {
if (request.readyState == 4) {
if (request.status == 200) {
var response = request.responseText.split("|");
document.getElementById("order").value = response[0];
document.getElementById("address").innerHTML =
response[1].replace(/\n/g, "");
} else
alert("status is " + request.status);
}
}
首 先,獲得 responseText 並使用 JavaScript split() 方法從管道符分開。獲得的數組放到 response 中。數組中的第一個值 —— 上一個訂單 —— 用 response[0] 訪問,被設置爲 ID 爲 「order」 的字段的值。第二個值 response[1],即客戶地址,則須要更多一點處理。由於地址中的行用通常的行分隔符(「\n」字符)分隔,代碼中須要用 XHTML 風格的行分隔符 <br /> 來代替。替換過程使用 replace() 函數和正則表達式完成。最後,修改後的文本做爲 HTML 表單 div 中的內部 HTML。結果就是表單忽然用客戶信息更新了,如圖 4 所示。
圖 4. 收到客戶數據後的 Break Neck 表單
結 束本文以前,我還要介紹 XMLHttpRequest 的另外一個重要屬性 responseXML。若是服務器選擇使用 XML 響應則該屬性包含(也許您已經猜到)XML 響應。處理 XML 響應和處理普通文本有很大不一樣,涉及到解析、文檔對象模型(DOM)和其餘一些問題。後面的文章中將進一步介紹 XML。可是由於 responseXML 一般和 responseText 一塊兒討論,這裏有必要提一提。對於不少簡單的 Ajax 應用程序 responseText 就夠了,可是您很快就會看到經過 Ajax 應用程序也能很好地處理 XML。
結束語
您可能 對 XMLHttpRequest 感到有點厭倦了,我不多看到一整篇文章討論一個對象,特別是這種簡單的對象。可是您將在使用 Ajax 編寫的每一個頁面和應用程序中反覆使用該對象。坦白地說,關於 XMLHttpRequest 還真有一些可說的內容。下一期文章中將介紹如何在請求中使用 POST 及 GET,來設置請求中的內容頭部和從服務器響應讀取內容頭部,理解如何在請求/響應模型中編碼請求和處理 XML。
再日後咱們將介紹常見 Ajax 工具箱。這些工具箱實際上隱藏了本文所述的不少細節,使得 Ajax 編程更容易。您也許會想,既然有這麼多工具箱爲什麼還要對底層的細節編碼。答案是,若是不知道應用程序在作什麼,就很難發現應用程序中的問題。
所以不要忽略這些細節或者簡單地瀏覽一下,若是便捷華麗的工具箱出現了錯誤,您就沒必要撓頭或者發送郵件請求支持了。若是瞭解如何直接使用 XMLHttpRequest,就會發現很容易調試和解決最奇怪的問題。只有讓其解決您的問題,工具箱纔是好東西。
所以請熟悉 XMLHttpRequest 吧。事實上,若是您有使用工具箱的 Ajax 代碼,能夠嘗試使用 XMLHttpRequest 對象及其屬性和方法從新改寫。這是一種不錯的練習,能夠幫助您更好地理解其中的原理。
下一期文章中將進一步討論該對象,探討它的一些更有趣的屬性(如 responseXML),以及如何使用 POST 請求和以不一樣的格式發送數據。請開始編寫代碼吧,一個月後咱們再繼續討論。
第 3 頁 Ajax 中的高級請求和響應
對 於不少 Web 開發人員來講,只須要生成簡單的請求並接收簡單的響應便可;可是對於但願掌握 Ajax 的開發人員來講,必需要全面理解 HTTP 狀態代碼、就緒狀態和 XMLHttpRequest 對象。在本文中,Brett McLaughlin 將向您介紹各類狀態代碼,並展現瀏覽器如何對其進行處理,本文還給出了在 Ajax 中使用的比較少見的 HTTP 請求。
在本系列的 上篇文章 中,咱們將詳細介紹 XMLHttpRequest 對象,它是 Ajax 應用程序的中心,負責處理服務器端應用程序和腳本的請求,並處理從服務器端組件返回的數據。因爲全部的 Ajax 應用程序都要使用 XMLHttpRequest 對象,所以您可能會但願熟悉這個對象,從而可以讓 Ajax 執行得更好。
在本文中,我將在上一篇文章的基礎上重點介紹這個請求對象的 3 個關鍵部分的內容:
·HTTP 就緒狀態
·HTTP 狀態代碼
·能夠生成的請求類型
這 三部份內容都是在構造一個請求時所要考慮的因素;可是介紹這些主題的內容太少了。然而,若是您不只僅是想了解 Ajax 編程的常識,而是但願瞭解更多內容,就須要熟悉就緒狀態、狀態代碼和請求自己的內容。當應用程序出現問題時 —— 這種問題老是存在 —— 那麼若是可以正確理解就緒狀態、如何生成一個 HEAD 請求或者 400 的狀態代碼的確切含義,就能夠在 5 分鐘內調試出問題,而不是在各類挫折和困惑中度過 5 個小時。
下面讓咱們首先來看一下 HTTP 就緒狀態。
深刻了解 HTTP 就緒狀態
您 應該還記得在上一篇文章中 XMLHttpRequest 對象有一個名爲 readyState 的屬性。這個屬性確保服務器已經完成了一個請求,一般會使用一個回調函數從服務器中讀出數據來更新 Web 表單或頁面的內容。清單 1 給出了一個簡單的例子(這也是本系列的上一篇文章中的一個例子 —— 請參見 參考資料)。
XMLHttpRequest 或 XMLHttp:換名玫瑰
Microsoft™ 和 Internet Explorer 使用了一個名爲 XMLHttp 的對象,而不是 XMLHttpRequest 對象,而 Mozilla、Opera、Safari 和 大部分非 Microsoft 瀏覽器都使用的是後者。爲了簡單性起見,我將這兩個對象都簡單地稱爲 XMLHttpRequest。這既符合咱們在 Web 上看到的狀況,又符合 Microsoft 在 Internet Explorer 7.0 中使用 XMLHttpRequest 做爲請求對象的意圖。(有關這個問題的更多內容,請參見 第 2 部分。)
清單 1. 在回調函數中處理服務器的響應
function updatePage() {
if (request.readyState == 4) {
if (request.status == 200) {
var response = request.responseText.split("|");
document.getElementById("order").value = response[0];
document.getElementById("address").innerHTML =
response[1].replace(/\n/g, "<br />");
} else
alert("status is " + request.status);
}
}
這顯然是就緒狀態最多見(也是最簡單)的用法。正如您從數字 "4" 中能夠看出的同樣,還有其餘幾個就緒狀態(您在上一篇文章中也看到過這個清單 —— 請參見 參考資料):
·0:請求未初始化(尚未調用 open())。
·1:請求已經創建,可是尚未發送(尚未調用 send())。
·2:請求已發送,正在處理中(一般如今能夠從響應中獲取內容頭)。
·3:請求在處理中;一般響應中已有部分數據可用了,可是服務器尚未完成響應的生成。
·4:響應已完成;您能夠獲取並使用服務器的響應了。
若是您但願不只僅是瞭解 Ajax 編程的基本知識,那麼就不但須要知道這些狀態,瞭解這些狀態是什麼時候出現的,以及如何來使用這些狀態。首先,您須要學習在每種就緒狀態下可能碰到的是哪一種請求狀態。不幸的是,這一點並不直觀,並且會涉及幾種特殊的狀況。
隱祕就緒狀態
第 一種就緒狀態的特色是 readyState 屬性爲 0(readyState == 0),表示未初始化狀態。一旦對請求對象調用 open() 以後,這個屬性就被設置爲 1。因爲您一般都是在一對請求進行初始化以後就當即調用 open(),所以不多會看到 readyState == 0 的狀態。另外,未初始化的就緒狀態在實際的應用程序中是沒有真正的用處的。
不過爲了知足咱們的興趣,請參見 清單 2 的內容,其中顯示瞭如何在 readyState 被設置爲 0 時來獲取這種就緒狀態。
清單 2. 獲取 0 就緒狀態
function getSalesData() {
// Create a request object
createRequest();
alert("Ready state is: " + request.readyState);
// Setup (initialize) the request
var url = "/boards/servlet/UpdateBoardSales";
request.open("GET", url, true);
request.onreadystatechange = updatePage;
request.send(null);
}
在這個簡單的例子中,getSalesData() 是 Web 頁面調用來啓動請求(例如點擊一個按鈕時)所使用的函數。注意您必須在調用 open()以前 來查看就緒狀態。圖 1 給出了運行這個應用程序的結果。
圖 1. 就緒狀態 0
顯 然,這並不能爲您帶來多少好處;須要確保 還沒有 調用 open() 函數的狀況不多。在大部分 Ajax 編程的真實狀況中,這種就緒狀態的惟一用法就是使用相同的 XMLHttpRequest 對象在多個函數之間生成多個請求。在這種(不常見的)狀況中,您可能會在生成新請求以前但願確保請求對象是處於未初始化狀態(readyState == 0)。這其實是要確保另一個函數沒有同時使用這個對象。
查看正在處理的請求的就緒狀態
除了 0 就緒狀態以外,請求對象還須要依次經歷典型的請求和響應的其餘幾種就緒狀態,最後才以就緒狀態 4 的形式結束。這就是爲何您在大部分回調函數中均可以看到 if (request.readyState == 4) 這行代碼;它確保服務器已經完成對請求的處理,如今能夠安全地更新 Web 頁面或根據從服務器返回來的數據來進行操做了。
要查看這種狀態發生的過程很是簡單。若是就緒狀態爲 4,咱們不只要運行回調函數中的代碼,並且還要在每次調用回調函數時都輸出就緒狀態。 清單 3 給出了一個實現這種功能的例子。
當 0 等於 4 時
在 多個 JavaScript 函數都使用相同的請求對象時,您須要檢查就緒狀態 0 來確保這個請求對象沒有正在使用,這種機制會產生問題。因爲 readyState == 4 表示一個已完成的請求,所以您常常會發現那些目前沒在使用的處於就緒狀態的請求對象仍然被設置成了 4 —— 這是由於從服務器返回來的數據已經使用過了,可是從它們被設置爲就緒狀態以後就沒有進行任何變化。有一個函數 abort() 會從新設置請求對象,可是這個函數卻不是真正爲了這個目的而使用的。若是您 必須 使用多個函數,最好是爲每一個函數都建立並使用一個函數,而不是在多個函數之間共享相同的對象。
清單 3. 查看就緒狀態
function updatePage() {
// Output the current ready state
alert("updatePage() called with ready state of " + request.readyState);
}
如 果您不肯定如何運行這個函數,就須要建立一個函數,而後在 Web 頁面中調用這個函數,並讓它向服務器端的組件發送一個請求(例如 清單 2 給出的函數,或本系列文章的第 1 部分和第 2 部分中給出的例子)。確保在創建請求時,將回調函數設置爲 updatePage();要實現這種設置,能夠將請求對象的 onreadystatechange 屬性設置爲 updatePage()。
這段代碼就是 onreadystatechange 意義的一個確切展現 —— 每次請求的就緒狀態發生變化時,就調用 updatePage(),而後咱們就能夠看到一個警告了。圖 2 給出了一個調用這個函數的例子,其中就緒狀態爲 1。
圖 2. 就緒狀態 1
您 能夠本身嘗試運行這段代碼。將其放入 Web 頁面中,而後激活事件處理程序(單擊按鈕,在域之間按 tab 鍵切換焦點,或者使用設置的任何方法來觸發請求)。這個回調函數會運行屢次 —— 每次就緒狀態都會改變 —— 您能夠看到每一個就緒狀態的警告。這是跟蹤請求所經歷的各個階段的最好方法。
瀏覽器的不一致性
在對這個過程有一個基本的瞭解以後,請試着從幾個不一樣的瀏覽器中訪問您的頁面。您應該會注意到各個瀏覽器如何處理這些就緒狀態並不一致。例如,在 Firefox 1.5 中,您會看到如下就緒狀態:
·1
·2
·3
·4
這並不奇怪,由於每一個請求狀態都在這裏表示出來了。然而,若是您使用 Safari 來訪問相同的應用程序,就應該看到 —— 或者看不到 —— 一些有趣的事情。下面是在 Safari 2.0.1 中看到的狀態:
·2
·3
·4
Safari 實際上把第一個就緒狀態給丟棄了,也並無什麼明顯的緣由說明爲何要這樣作;不過這就是 Safari 的工做方式。這還說明了一個重要的問題:儘管在使用服務器上的數據以前確保請求的狀態爲 4 是一個好主意,可是依賴於每一個過渡期就緒狀態編寫的代碼的確會在不一樣的瀏覽器上獲得不一樣的結果。
例如,在使用 Opera 8.5 時,所顯示的就緒狀態狀況就更加糟糕了:
·3
·4
最後,Internet Explorer 會顯示以下狀態:
·1
·2
·3
·4
若是您碰到請求方面的問題,這就是用來發現問題的 首要之處。最好的方式是在 Internet Explorer 和 Firefox 都進行一下測試 —— 您會看到全部這 4 種狀態,並能夠檢查請求的每一個狀態所處的狀況。
接下來咱們再來看一下響應端的狀況。
顯微鏡下的響應數據
一 旦咱們理解在請求過程當中發生的各個就緒狀態以後,接下來就能夠來看一下 XMLHttpRequest 對象的另一個方面了 —— responseText 屬性。回想一下在上一篇文章中咱們介紹過的內容,就能夠知道這個屬性用來從服務器上獲取數據。一旦服務器完成對請求的處理以後,就能夠將響應請求數據所需 要的任何數據放到請求的 responseText 中了。而後回調函數就能夠使用這些數據,如 清單 1 和 清單 4 所示。
清單 4. 使用服務器上返回的響應
function updatePage() {
if (request.readyState == 4) {
var newTotal = request.responseText;
var totalSoldEl = document.getElementById("total-sold");
var netProfitEl = document.getElementById("net-profit");
replaceText(totalSoldEl, newTotal);
/* 圖 out the new net profit */
var boardCostEl = document.getElementById("board-cost");
var boardCost = getText(boardCostEl);
var manCostEl = document.getElementById("man-cost");
var manCost = getText(manCostEl);
var profitPerBoard = boardCost - manCost;
var netProfit = profitPerBoard * newTotal;
/* Update the net profit on the sales form */
netProfit = Math.round(netProfit * 100) / 100;
replaceText(netProfitEl, netProfit);
}
清單 1 至關簡單;清單 4 稍微有點複雜,可是它們在開始時都要檢查就緒狀態,並獲取 responseText 屬性的值。
查看請求的響應文本
與就緒狀態相似,responseText 屬性的值在整個請求的生命週期中也會發生變化。要查看這種變化,請使用如 清單 5 所示的代碼來測試請求的響應文本,以及它們的就緒狀態。
清單 5. 測試 responseText 屬性
function updatePage() {
// Output the current ready state
alert("updatePage() called with ready state of " + request.readyState +
" and a response text of '" + request.responseText + "'");
}
現 在在瀏覽器中打開 Web 應用程序,並激活您的請求。要更好地看到這段代碼的效果,請使用 Firefox 或 Internet Explorer,由於這兩個瀏覽器均可以報告出請求過程當中全部可能的就緒狀態。例如在就緒狀態 2 中,就沒有定義 responseText (請參見 圖 3);若是 JavaScript 控制檯也已經打開了,您就會看到一個錯誤。
圖 3. 就緒狀態爲 2 的響應文本
不過在就緒狀態 3 中,服務器已經在 responseText 屬性中放上了一個值,至少在這個例子中是這樣(請參見 圖 4)。
圖 4. 就緒狀態爲 3 的響應文本
您會看到就緒狀態爲 3 的響應在每一個腳本、每一個服務器甚至每一個瀏覽器上都是不同的。不過,這在調試應用程序中依然是很是有用的。
獲取安全數據
所 有的文檔和規範都強調,只有在就緒狀態爲 4 時數據才能夠安全使用。相信我,當就緒狀態爲 3 時,您不多能找到沒法從 responseText 屬性獲取數據的狀況。然而,在應用程序中將本身的邏輯依賴於就緒狀態 3 可不是什麼好主意 —— 一旦您編寫了依賴於就緒狀態 3 的完整數據的的代碼,幾乎就要本身來負責當時的數據不完整問題了。
比較好的作法是向用戶提供一些反饋,說明在處於就緒狀態 3 時,很快就會有響應了。儘管使用 alert() 之類的函數顯然不是什麼好主意 —— 使用 Ajax 而後使用一個警告對話框來阻塞用戶顯然是錯誤的 —— 不過您能夠在就緒狀態發生變化時更新表單或頁面中的域。例如,對於就緒狀態 1 來講要將進度指示器的寬度設置爲 25%,對於就緒狀態 2 來講要將進度指示器的寬度設置爲 50%,對於就緒狀態 3 來講要將進度指示器的寬度設置爲 75%,當就緒狀態爲 4 時將進度指示器的寬度設置爲 100%(完成)。
固然,正如您已經看到的同樣,這種方法很是聰明,但它是依賴於瀏覽器的。在 Opera 上,您永遠都不會看到前兩個就緒狀態,而在 Safari 上則沒有第一個(1)。因爲這個緣由,我將這段代碼留做練習,而沒有在本文中包括進來。
如今應該來看一下狀態代碼了。
深刻了解 HTTP 狀態代碼
有 了就緒狀態和您在 Ajax 編程技術中學習到的服務器的響應,您就能夠爲 Ajax 應用程序添加另一級複雜性了 —— 這要使用 HTTP 狀態代碼。這些代碼對於 Ajax 來講並無什麼新鮮。從 Web 出現以來,它們就已經存在了。在 Web 瀏覽器中您可能已經看到過幾個狀態代碼:
·401:未經受權
·403:禁止
·404:沒找到
您能夠找到更多的狀態代碼(完整清單請參見 參考資料)。要爲 Ajax 應用程序另外添加一層控制和響應(以及更爲健壯的錯誤處理)機制,您須要適當地查看請求和響應中的狀態代碼。
200:一切正常
在不少 Ajax 應用程序中,您將看到一個回調函數,它負責檢查就緒狀態,而後繼續利用從服務器響應中返回的數據,如 清單 6 所示。
清單 6. 忽略狀態代碼的回調函數
function updatePage() {
if (request.readyState == 4) {
var response = request.responseText.split("|");
document.getElementById("order").value = response[0];
document.getElementById("address").innerHTML =
response[1].replace(/\n/g, "<br />");
}
}
這 對於 Ajax 編程來講證實是一種短視而錯誤的方法。若是腳本須要認證,而請求卻沒有提供有效的證書,那麼服務器就會返回諸如 403 或 401 之類的錯誤代碼。然而,因爲服務器對請求進行了應答,所以就緒狀態就被設置爲 4(即便應答並非請求所指望的也是如此)。最終,用戶沒有得到有效數據,當 JavaScript 試圖使用不存在的服務器數據時就可能會出現嚴重的錯誤。
它花費了最小的努力來確保服務器不但完成了一個請求,並且還返回了一個 「一切良好」 的狀態代碼。這個代碼是 "200",它是經過 XMLHttpRequest 對象的 status 屬性來報告的。爲了確保服務器不但完成了一個請求,並且還報告了一個 OK 狀態,請在您的回調函數中添加另一個檢查功能,如 清單 7 所示。
清單 7. 檢查有效狀態代碼
function updatePage() {
if (request.readyState == 4) {
if (request.status == 200) {
var response = request.responseText.split("|");
document.getElementById("order").value = response[0];
document.getElementById("address").innerHTML =
response[1].replace(/\n/g, "<br />");
} else
alert("status is " + request.status);
}
}
經過添加這幾行代碼,您就能夠確認是否存在問題,用戶會看到一個有用的錯誤消息,而不只僅是看到一個由斷章取義的數據所構成的頁面,而沒有任何解釋。
重定向和從新路由
在深刻介紹有關錯誤的內容以前,咱們有必要來討論一下有關一個在使用 Ajax 時 並不須要 關心的問題 —— 重定向。在 HTTP 狀態代碼中,這是 300 系列的狀態代碼,包括:
·301:永久移動
·302:找到(請求被從新定向到另一個 URL/URI 上)
·305:使用代理(請求必須使用一個代理來訪問所請求的資源)
Ajax 程序員可能並不太關心有關重定向的問題,這是因爲兩方面的緣由:
· 首先,Ajax 應用程序一般都是爲一個特定的服務器端腳本、servlet 或應用程序而編寫的。對於那些您看不到就消失了的組件來講,Ajax 程序員就不太清楚了。所以有時您會知道資源已經移動了(由於您移動了它,或者經過某種手段移動了它),接下來要修改請求中的 URL,而且不會再碰到這種結果了。
更爲重要的一個緣由是:Ajax 應用程序和請求都是封裝在沙盒中的。這就意味着提供生成 Ajax 請求的 Web 頁面的域必須是對這些請求進行響應的域。所以 ebay.com 所提供的 Web 頁面就不能對一個在 amazon.com 上運行的腳本生成一個 Ajax 風格的請求;在 ibm.com 上的 Ajax 應用程序也沒法對在 netbeans.org 上運行的 servlets 發出請求。
·結果是您的請求沒法重定向到其餘服務器上,而不會產生安全性錯誤。在這些狀況中,您根本就不會獲得狀態代碼。一般在調試控制檯中都會產生一個 JavaScript 錯誤。所以,在對狀態代碼進行充分的考慮以後,您就能夠徹底忽略重定向代碼的問題了。
結果是您的請求沒法重定向到其餘服務器上,而不會產生安全性錯誤。在這些狀況中,您根本就不會獲得狀態代碼。一般在調試控制檯中都會產生一個 JavaScript 錯誤。所以,在對狀態代碼進行充分的考慮以後,您就能夠徹底忽略重定向代碼的問題了。
錯誤
一 旦接收到狀態代碼 200 而且意識到能夠很大程度上忽略 300 系列的狀態代碼以後,所須要擔憂的惟一一組代碼就是 400 系列的代碼了,這說明了不一樣類型的錯誤。回頭再來看一下 清單 7,並注意在對錯誤進行處理時,只將少數常見的錯誤消息輸出給用戶了。儘管這是朝正確方向前進的一步,可是要告訴從事應用程序開發的用戶和程序員究竟發生 了什麼問題,這些消息仍然是沒有太大用處的。
首先,咱們要添加對找不到的頁的支持。實際上這在大部分產品系統中都不該該出現,可是在測試 腳本位置發生變化或程序員輸入了錯誤的 URL 時,這種狀況並不罕見。若是您能夠天然地報告 404 錯誤,就能夠爲那些困擾不堪的用戶和程序員提供更多幫助。例如,若是服務器上的一個腳本被刪除了,咱們就能夠使用 清單 7 中的代碼,這樣用戶就會看到一個如 圖 5 所示的非描述性錯誤。
邊界狀況和困難狀況
看到如今,一些新手程序員就可能會 這到底是要討論什麼內容。有一點事實你們須要知道:只有不到 5% 的 Ajax 請求須要使用諸如 二、3 之類的就緒狀態和諸如 403 之類的狀態代碼(實際上,這個比率可能更接近於 1% 甚至更少)。這些狀況很是重要,稱爲 邊界狀況(edge case) —— 它們只會在一些很是特殊的狀況下發生,其中遇到的都是最奇特的問題。雖然這些狀況並不廣泛,可是這些邊界狀況卻佔據了大部分用戶所碰到的問題的 80%!
對於典型的用戶來講,應用程序 100 次都是正常工做的這個事實一般都會被忘記,然而應用程序只要一次出錯就會被他們清楚地記住。若是您能夠很好地處理邊界狀況(或困難狀況),就能夠爲再次訪問站點的用戶提供滿意的回報。
圖 5. 常見錯誤處理
用戶沒法判斷問題到底是認證問題、沒找到腳本(此處就是這種狀況)、用戶錯誤仍是代碼中有些地方產生了問題。添加一些簡單的代碼可讓這個錯誤更加具體。請參照 清單 8,它負責處理沒找到的腳本或認證發生錯誤的狀況,在出現這些錯誤時都會給出具體的消息。
清單 8. 檢查有效狀態代碼
function updatePage() {
if (request.readyState == 4) {
if (request.status == 200) {
var response = request.responseText.split("|");
document.getElementById("order").value = response[0];
document.getElementById("address").innerHTML =
response[1].replace(/\n/g, "<br />");
} else if (request.status == 404) {
alert ("Requested URL is not found.");
} else if (request.status == 403) {
alert("Access denied.");
} else
alert("status is " + request.status);
}
}
雖然這依然至關簡單,可是它的確多提供了一些有用的信息。圖 6 給出了與 圖 5 相同的錯誤,可是這一次錯誤處理代碼向用戶或程序員更好地說明了究竟發生了什麼。
圖 6. 特殊錯誤處理
在 咱們本身的應用程序中,能夠考慮在發生認證失敗的狀況時清除用戶名和密碼,並向屏幕上添加一條錯誤消息。咱們能夠使用相似的方法來更好地處理找不到腳本或 其餘 400 類型的錯誤(例如 405 表示不容許使用諸如發送 HEAD 請求之類不可接受的請求方法,而 407 則表示須要進行代理認證)。然而無論採用哪一種選擇,都須要從對服務器上返回的狀態代碼開始入手進行處理。
其餘請求類型
如 果您真但願控制 XMLHttpRequest 對象,能夠考慮最後實現這種功能 —— 將 HEAD 請求添加到指令中。在前兩篇文章中,咱們已經介紹瞭如何生成 GET 請求;在立刻就要發表的一篇文章中,您會學習有關使用 POST 請求將數據發送到服務器上的知識。不過本着加強錯誤處理和信息蒐集的精神,您應該學習如何生成 HEAD 請求。
生成請求
實際上生成 HEAD 請求很是簡單;您能夠使用 "HEAD"(而不是 "GET" 或 "POST")做爲第一個參數來調用 open() 方法,如 清單 9 所示。
清單 9. 使用 Ajax 生成一個 HEAD 請求
function getSalesData() {
createRequest();
var url = "/boards/servlet/UpdateBoardSales";
request.open("HEAD", url, true);
request.onreadystatechange = updatePage;
request.send(null);
}
當 您這樣生成一個 HEAD 請求時,服務器並不會像對 GET 或 POST 請求同樣返回一個真正的響應。相反,服務器只會返回資源的 頭(header),這包括響應中內容最後修改的時間、請求資源是否存在和不少其餘有用信息。您能夠在服務器處理並返回資源以前使用這些信息來了解有關資 源的信息。
對於這種請求您能夠作的最簡單的事情就是簡單地輸出全部的響應頭的內容。這可讓您瞭解經過 HEAD 請求能夠使用什麼。清單 10 提供了一個簡單的回調函數,用來輸出從 HEAD 請求中得到的響應頭的內容。
清單 10. 輸出從 HEAD 請求中得到的響應頭的內容
function updatePage() {
if (request.readyState == 4) {
alert(request.getAllResponseHeaders());
}
}
請參見 圖 7,其中顯示了從一個向服務器發出的 HEAD 請求的簡單 Ajax 應用程序返回的響應頭。
您能夠單獨使用這些頭(從服務器類型到內容類型)在 Ajax 應用程序中提供其餘信息或功能。
檢查 URL
您 已經看到了當 URL 不存在時應該如何檢查 404 錯誤。若是這變成一個常見的問題 —— 多是缺乏了一個特定的腳本或 servlet —— 那麼您就可能會但願在生成完整的 GET 或 POST 請求以前來檢查這個 URL。要實現這種功能,生成一個 HEAD 請求,而後在回調函數中檢查 404 錯誤;清單 11 給出了一個簡單的回調函數。
清單 11. 檢查某個 URL 是否存在
function updatePage() {
if (request.readyState == 4) {
if (request.status == 200) {
alert("URL exists");
} else if (request.status == 404) {
alert("URL does not exist.");
} else {
alert("Status is: " + request.status);
}
}
}
誠 實地說,這段代碼的價值並不太大。服務器必須對請求進行響應,並構造一個響應來填充內容長度的響應頭,所以並不能節省任何處理時間。另外,這花費的時間與 生成請求並使用 HEAD 請求來查看 URL 是否存在所須要的時間同樣多,由於它要生成使用 GET 或 POST 的請求,而不只僅是如 清單 7 所示同樣來處理錯誤代碼。不過,有時確切地瞭解目前什麼可用也是很是有用的;您永遠不會知道什麼時候創造力就會迸發或者什麼時候須要 HEAD 請求!
有用的 HEAD 請求
您 會發現 HEAD 請求很是有用的一個領域是用來查看內容的長度或內容的類型。這樣能夠肯定是否須要發回大量數據來處理請求,和服務器是否試圖返回二進制數據,而不是 HTML、文本或 XML(在 JavaScript 中,這 3 種類型的數據都比二進制數據更容易處理)。
在這些狀況中,您只使用了 適當的頭名,並將其傳遞給 XMLHttpRequest 對象的 getResponseHeader() 方法。所以要獲取響應的長度,只須要調用 request.getResponseHeader("Content-Length");。要獲取內容類型,請使用 request.getResponseHeader("Content-Type");。
在不少應用程序中,生成 HEAD 請求並無增長任何功能,甚至可能會致使請求速度變慢(經過強制生成一個 HEAD 請求來獲取有關響應的數據,而後在使用一個 GET 或 POST 請求來真正獲取響應)。然而,在出現您不肯定有關腳本或服務器端組件的狀況時,使用 HEAD 請求能夠獲取一些基本的數據,而不須要對響應數據真正進行處理,也不須要大量的帶寬來發送響應。
結束語
對於不少 Ajax 和 Web 程序員來講,本文中介紹的內容彷佛是過高級了。生成 HEAD 請求的價值是什麼呢?到底在什麼狀況下須要在 JavaScript 中顯式地處理重定向狀態代碼呢?這些都是很好的問題;對於簡單的應用程序來講,答案是這些高級技術的價值並非很是大。
然而,Web 已經再也不是隻需實現簡單應用程序的地方了;用戶已經變得更加高級,客戶指望可以得到更好的穩定性、更高級的錯誤報告,若是應用程序有 1% 的時間停機,那麼經理就可能會所以而被解僱。
所以您的工做就不能僅僅侷限於簡單的應用程序了,而是須要更深刻理解 XMLHttpRequest。
·若是您能夠考慮各類就緒狀態 —— 而且理解了這些就緒狀態在不一樣瀏覽器之間的區別 —— 就能夠快速調試應用程序了。您甚至能夠基於就緒狀態而開發一些創造性的功能,並向用戶和客戶回報請求的狀態。
·若是您要對狀態代碼進行控制,就能夠設置應用程序來處理腳本錯誤、非預期的響應以及邊緣狀況。結果是應用程序在全部的時間均可以正常工做,而不只僅是隻能一切都正常的狀況下才能運行。
·增長這種生成 HEAD 請求的能力,檢查某個 URL 是否存在,以及確認某個文件是否被修改過,這樣就能夠確保用戶能夠得到有效的頁面,用戶所看到的信息都是最新的,(最重要的是)讓他們驚訝這個應用程序是如何健壯和通用。
本 文的目的並不是是要讓您的應用程序顯得十分華麗,而是幫助您去掉黃色聚光燈後重點昭顯文字的美麗,或者外觀更像桌面同樣。儘管這些都是 Ajax 的功能(在後續幾篇文章中就會介紹),不過它們卻像是蛋糕表面的一層奶油。若是您能夠使用 Ajax 來構建一個堅實的基礎,讓應用程序能夠很好地處理錯誤和問題,用戶就會返回您的站點和應用程序。在接下來的文章中,咱們將添加這種直觀的技巧,這會讓客戶 興奮得發抖。前端
第 4 頁 利用 DOM 進行 Web 響應
程 序員(使用後端應用程序)和 Web 程序員(編寫 HTML、CSS 和 JavaScript)之間的分水嶺是長久存在的。可是,Document Object Model (DOM) 彌補了這個裂縫,使得在後端使用 XML 同時在前端使用 HTML 切實可行,併成爲極其有效的工具。在本文中,Brett McLaughlin 介紹了 Document Object Model,解釋它在 Web 頁面中的應用,並開始挖掘其在 JavaScript 中的用途。
與許多 Web 程序員同樣,您可能使用過 HTML。HTML 是程序員開始與 Web 頁面打交道的方式;HTML 一般是他們完成應用程序或站點前的最後一步——調整一些佈局、顏色或樣式。不過,雖然常用 HTML,但對於 HTML 轉到瀏覽器呈如今屏幕上時到底發生了什麼,人們廣泛存在誤解。在我分析您認爲可能發生的事情及其可能錯誤的緣由以前,我但願您對設計和服務 Web 頁面時涉及的過程一清二楚:
一、一些人(一般是您!)在文本編輯器或 IDE 中建立 HTML。
二、而後您將 HTML 上載到 Web 服務器,例如 Apache HTTPD,並將其公開在 Internet 或 intranet 上。
三、用戶用 Firefox 或 SafariA 等瀏覽器請求您的 Web 頁面。
四、用戶的瀏覽器向您的服務器請求 HTML。
五、瀏覽器將從服務器接收到的頁面以圖形和文本方式呈現;用戶看到並激活 Web 頁面。
這看起來很是基礎,但事情很快會變得有趣起來。事實上,步驟 4 和步驟 5 之間發生的巨大數量的 「填充物(stuff)」 就是本文的焦點。術語 「填充物」 也十分適用,由於多數程序員歷來沒有真正考慮過當用戶瀏覽器請求顯示標記時到底在標記身上發生了什麼。
·是否瀏覽器只是讀取 HTML 中的文本並將其顯示?
·CSS 呢?尤爲是當 CSS 位於外部文件時。
·JavaScript 呢?它也一般位於外部文件中。
·瀏覽器如何處理這些項,若是將事件處理程序、函數和樣式映射到該文本標記?
實踐證實,全部這些問題的答案都是 Document Object Model。所以,廢話少說,直接研究 DOM。
Web 程序員和標記
對 於多數程序員,當 Web 瀏覽器開始時他們的工做就結束了。也就是說,將一個 HTML 文件放入 Web 瀏覽器的目錄上後,您一般就認爲它已經「完成」,並且(滿懷但願地)認爲不再會考慮它!說到編寫乾淨、組織良好的頁面時,這也是一個偉大的目標;但願您 的標記跨瀏覽器、用各類版本的 CSS 和 JavaScript 顯示它應該顯示的內容,一點錯都沒有。
問題是這種方法限制了程序員對瀏覽器中真正發生的事情的理解。更重要的是,它限制了您用客戶端 JavaScript 動態更新、更改和重構 Web 頁面的能力。擺脫這種限制,讓您的 Web 站點擁有更大的交互性和創造性。
程序員作什麼
做 爲典型的 Web 程序員,您可能啓動文本編輯和 IDE 後就開始輸入 HTML、CSS 甚至 JavaScript。很容易認爲這些標記、選擇器和屬性只是使站點正確顯示而作的小小的任務。可是,在這一點上您須要拓展您的思路,要意識到您是在組織 您的內容。不要擔憂;我保證這不會變成關於標記美觀、您必須如何認識到 Web 頁面的真正潛力或其餘任何元物質的講座。您須要瞭解的是您在 Web 開發中究竟是什麼角色。
說到頁面的外觀,頂多您只能提提建議。您提供 CSS 樣式表時,用戶能夠覆蓋您的樣式選擇。您提供字體大小時,用戶瀏覽器能夠爲視障者更改這些大小,或者在大顯示器(具備同等大的分辨率)上按比例縮小。甚至 您選擇的顏色和字體也受制於用戶顯示器和用戶在其系統上安裝的字體。雖然盡您所能來設計頁面樣式很不錯,但這毫不是 您對 Web 頁面的最大影響。
您 絕對控制的是 Web 頁面的結構。您的標記不可更改,用戶就不能亂弄;他們的瀏覽器只能從您的 Web 服務器檢索標記並顯示它(雖然樣式更符合用戶的品味而不是您本身的品味)。但頁面組織,無論是在該段落內仍是在其餘分區,都只由您單獨決定。要是想實際更 改您的頁面(這是大多數 Ajax 應用程序所關注的),您操做的是頁面的結構。儘管很容易更改一段文本的顏色,但在現有頁面上添加文本或整個區段要可貴多。無論用戶如何設計該區段的樣式, 都是由您控制頁面自己的組織。
標記作什麼
一旦意識到您的標記是真正與組織相關的,您就會對它刮目相看了。不會認爲 h1 致使文本是大字號、黑色、粗體的,而會認爲 h1 是標題。用戶如何看待這個問題以及他們是使用您的 CSS、他們本身的 CSS 仍是這二者的組合,這是次要的考慮事項。相反,要意識到只有標記才能提供這種級別的組織;p 指明文本在段落內,img 表示圖像,div 將頁面分紅區段,等等。
還應該清楚,樣式和行爲(事件處理程序和 JavaScript)是在過後 應用於該組織的。標記就緒之後才能對其進行操做或設計樣式。因此,正如您能夠將 CSS 保存在 HTML 的外部文件中同樣,標記的組織與其樣式、格式和行爲是分離的。雖然您確定能夠用 JavaScript 更改元素或文本的樣式,但實際更改您的標記所佈置的組織卻更加有趣。
只要牢記您的標記只爲您的頁面提供組織、框架,您就能立於不敗之地。再前進一小步,您就會明白瀏覽器是如何接受全部的文本組織並將其轉變爲超級有趣的一些東西的,即一組對象,其中每一個對象均可被更改、添加或刪除。
文本標記的優勢
在 討論 Web 瀏覽器以前,值得考慮一下爲何純文本絕對 是存儲 HTML 的最佳選擇(有關詳細信息,請參閱 有關標記的一些其餘想法)。不考慮優缺點,只是回憶一下在每次查看頁面時 HTML 是經過網絡發送到 Web 瀏覽器的(爲了簡潔,不考慮高速緩存等)。真是再沒有比傳遞文本再有效的方法了。二進制對象、頁面圖形表示、從新組織的標記塊等等,全部這一切都比純文本 文件經過網絡傳遞要更困難。
此外,瀏覽器也爲此增光添彩。今天的瀏覽器容許用戶更改文本大小、按比例伸縮圖像、下載頁面的 CSS 或 JavaScript(大多數狀況),甚至更多,這徹底排除了將任何類型的頁面圖形表示發送到瀏覽器上。可是,瀏覽器須要原 HTML,這樣它才能在瀏覽器中對頁面應用任何處理,而不是信任瀏覽器去處理該任務。一樣地,將 CSS 從 JavaScript 分離和將 CSS 從 HTML 標記分離要求一種容易分離的格式。文本文件又一次成爲該任務的最好方法。
最後但一樣重要的一點是,記住,新標準(好比 HTML 4.01 與 XHTML 1.0 和 1.1)承諾將內容(頁面中的數據)與表示和樣式(一般由 CSS 應用)分離。若是程序員要將 HTML 與 CSS 分離,而後強制瀏覽器檢索粘結頁面各部分的一些頁面表示,這會失去這些標準的多數優勢。保持這些部分到達瀏覽器時都一直分離使得瀏覽器在從服務器獲取 HTML 時有了史無前例的靈活性。
關於標記的其餘想法
純文本編輯:是對是錯?
純文本是存儲標記的理想選擇,但 是不適合編輯 標記。大行其道的是使用 IDE,好比 Macromedia DreamWeaver 或更強勢點的 Microsoft® FrontPage®,來操做 Web 頁面標記。這些環境一般提供快捷方式和幫助來建立 Web 頁面,尤爲是在使用 CSS 和 JavaScript 時,兩者都來自實際頁面標記之外的文件。許多人仍偏心好用古老的記事本或 vi(我認可我也是其中一員),這並沒關係。無論怎樣,最終結果都是充滿標記的文本文件。
網絡上的文本:好東西
已經說過,文本是文檔的最好媒體,好比 HTML 或 CSS,在網絡上被千百次地傳輸。當我說瀏覽器表示文本很難時,是特指將文本轉換爲用戶查看的可視圖形頁面。這與瀏覽器實際上如何從 Web 瀏覽器檢索頁面沒有關係;在這種狀況下,文本仍然是最佳選擇。
文本標記的缺點
正如文本標記對於設計人員和頁面建立者具備驚人的優勢以外,它對於瀏覽器也具備至關出奇的缺點。具體來講,瀏覽器很難直接將文本標記可視地表示給用戶(詳細信息請參閱 有關標記的一些其餘想法)。考慮下列常見的瀏覽器任務:
·基於元素類型、類、ID 及其在 HTML 文檔中的位置,將 CSS 樣式(一般來自外部文件中的多個樣式表)應用於標記。
·基於 JavaScript 代碼(一般位於外部文件)將樣式和格式應用於 HTML 文檔的不一樣部分。
·基於 JavaScript 代碼更改表單字段的值。
·基於 JavaScript 代碼,支持可視效果,好比圖像翻轉和圖像交換。
複雜性並不在於編碼這些任務;其中每件事都是至關容易的。複雜性來自實際實現請求動做的瀏覽器。若是標記存儲爲文本,好比,想要在 center-text 類的 p 元素中輸入文本 (text-align: center),如何實現呢?
·將內聯樣式添加到文本嗎?
·將樣式應用到瀏覽器中的 HTML 文本,並只保持內容居中或不居中?
·應用無樣式的 HTML,而後過後應用格式?
這些很是困難的問題是現在不多有人編寫瀏覽器的緣由。(編寫瀏覽器的人應該接受最由衷的感謝)
無疑,純文本不是存儲瀏覽器 HTML 的好辦法,儘管文本是獲取頁面標記最好的解決方案。若是加上 JavaScript 更改 頁面結構的能力,事情就變得有些微妙了。瀏覽器應該將修改過的結構從新寫入磁盤嗎?如何才能保持文檔的最新版本呢?
無疑,文本不是答案。它難以修改,爲其應用樣式和行爲很困難,與今天 Web 頁面的動態本質在根本上相去甚遠。
求助於樹視圖
這個問題的答案(至少是由當今 Web 瀏覽器選擇的答案)是使用樹結構來表示 HTML。參見 清單 1,這是一個表示爲本文標記的至關簡單又無聊的 HTML 頁面。
清單 1. 文本標記中的簡單 HTML 頁面
<html>
<head>
<title>Trees, trees, everywhere</title>
</head>
<body>
<h1>Trees, trees, everywhere</h1>
<p>Welcome to a <em>really</em> boring page.</p>
<div>
Come again soon.
<img src="come-again.gif" />
</div>
</body>
</html>
瀏覽器接受該頁面並將之轉換爲樹形結構,如圖 1 所示。
爲 了保持本文的進度,我作了少量簡化。DOM 或 XML 方面的專家會意識到空白對於文檔文本在 Web 瀏覽器樹結構中表示和分解方式的影響。膚淺的瞭解只會使事情變得模棱兩可,因此若是想弄清空白的影響,那最好不過了;若是不想的話,那能夠繼續讀下去,不 要考慮它。當它成爲問題時,那時您就會明白您須要的一切。
除了實際的樹背景以外,可能會首先注意到樹中的一切是以最外層的 HTML 包含元素,即 html 元素開始的。使用樹的比喻,這叫作根元素。因此即便這是樹的底層,當您查看並分析樹的時候,我也一般以此開始。若是它確實奏效,您能夠將整個樹顛倒一下, 但這確實有些拓展了樹的比喻。
從根流出的線表示不一樣標記部分之間的關係。head 和 body 元素是 html 根元素的孩子;title 是 head 的孩子,而文本 「Trees, trees, everywhere」 是 title 的孩子。整個樹就這樣組織下去,直到瀏覽器得到與 圖 1 相似的結構。
一些附加術語
爲了沿用樹的比喻,head 和 body 被叫作 html 的分支(branches)。叫分支是由於它們有本身的孩子。當到達樹的末端時,您將進入主要的文本,好比 「Trees, trees, everywhere」 和 「really」;這些一般稱爲葉子,由於它們沒有本身的孩子。您不須要記住全部這些術語,當您試圖弄清楚特定術語的意思時,只要想像一下樹結構就容易多 了。
對象的值
既然瞭解了一些基本的術語,如今應該關注一下其中包含元素名稱和文本的小矩形了(圖 1)。每一個矩形是一個對象;瀏覽器在其中解決一些文本問題。經過使用對象來表示 HTML 文檔的每一部分,能夠很容易地更改組織、應用樣式、容許 JavaScript 訪問文檔,等等。
對象類型和屬性
標記的每一個可能類型都有本身的對象類型。例如,HTML 中的元素用 Element 對象類型表示。文檔中的文本用 Text 類型表示,屬性用 Attribute 類型表示,以此類推。
所 以 Web 瀏覽器不只能夠使用對象模型來表示文檔(從而避免了處理靜態文本),還能夠用對象類型當即辨別出某事物是什麼。HTML 文檔被解析並轉換爲對象集合,如 圖 1 所示,而後尖括號和轉義序列(例如,使用 < 表示 <,使用 > 表示 >)等事物再也不是問題了。這就使得瀏覽器的工做(至少在解析輸入 HTML 以後)變得更容易。弄清某事物到底是元素仍是屬性並肯定如何處理該類型的對象,這些操做都十分簡單了。
經過使用對象,Web 瀏覽器能夠更改這些對象的屬性。例如,每一個元素對象具備一個父元素和一系列子元素。因此添加新的子元素或文本只須要向元素的子元素列表中添加一個新的子元 素。這些對象還具備 style 屬性,因此快速更改元素或文本段的樣式很是簡單。例如,要使用 JavaScript 更改 div 的高度,以下所示:
someDiv.style.height = "300px";
換句話說,Web 瀏覽器使用對象屬性能夠很是容易地更改樹的外觀和結構。將之比做瀏覽器在內部將頁面表示爲文本時必須進行的復瑣事情,每次更改屬性或結構都須要瀏覽器從新編寫靜態文件、從新解析並在屏幕上從新顯示。有了對象,全部這一切都解決了。
如今,花點時間展開一些 HTML 文檔並用樹將其勾畫出來。儘管這看起來是個不尋常的請求(尤爲是在包含極少代碼的這樣一篇文章中),若是您但願可以操縱這些樹,那麼須要熟悉它們的結構。
在這個過程當中,可能會發現一些古怪的事情。好比,考慮下列狀況:
·屬性發生了什麼?
·分解爲元素(好比 em 和 b)的文本呢?
·結構不正確(好比當缺乏結束 p 標記時)的 HTML 呢?
一旦熟悉這些問題以後,就能更好地理解下面幾節了。
嚴格有時是好事
如 果嘗試剛提到的練習 I,您可能會發現標記的樹視圖中存在一些潛在問題(若是不練習的話,那就聽我說吧!)。事實上,在 清單 1 和 圖 1 中就會發現一些問題,首先看 p 元素是如何分解的。若是您問一般的 Web 開發人員 「p 元素的文本內容是什麼」,最多見的答案將是 「Welcome to a really boring Web page.」。若是將之與圖 1 作比較,將會發現這個答案(雖然合乎邏輯)是根本不正確的。
實際上,p 元素具備三個 不一樣的子對象,其中沒有一個包含完整的 「Welcome to a really boring Web page.」 文本。您會發現文本的一部分,好比 「Welcome to a 」 和 「 boring Web page」,但不是所有。爲了理解這一點,記住標記中的任何內容都必須轉換爲某種類型的對象。
此外,順序可有可無!若是瀏覽器顯示正確的對象,但顯示順序與您在 HTML 中提供的順序不一樣,那麼您能想像出用戶將如何響應 Web 瀏覽器嗎?段落夾在頁面標題和文章標題中間,而這不是您本身組織文檔時的樣式呢?很顯然,瀏覽器必須保持元素和文本的順序。
在本例中,p 元素有三個不一樣部分:
·em 元素以前的文本
·em 元素自己
·em 元素以後的文本
若是將該順序打亂,可能會把重點放在文本的錯誤部分。爲了保持一切正常,p 元素有三個子對象,其順序是在 清單 1 的 HTML 中顯示的順序。並且,重點文本 「really」 不是p 的子元素;而是 p 的子元素 em 的子元素。
理解這一律念很是重要。儘管 「really」 文本將可能與其餘 p 元素文本一塊兒顯示,但它還是 em 元素的直接子元素。它能夠具備與其餘 p 文本不一樣的格式,並且能夠獨立於其餘文本處處移動。
要將之牢記在心,試着用圖表示清單 2 和 3 中的 HTML,確保文本具備正確的父元素(而無論文本最終會如何顯示在屏幕上)。
清單 2. 帶有巧妙元素嵌套的標記
<html>
<head>
<title>This is a little tricky</title>
</head>
<body>
<h1>Pay <u>close</u> attention, OK?</h1>
<div>
<p>This p really isn't <em>necessary</em>, but it makes the
<span id="bold-text">structure <i>and</i> the organization</span>
of the page easier to keep up with.</p>
</div>
</body>
</html>
清單 3. 更巧妙的元素嵌套
<html>
<head>
<title>Trickier nesting, still</title>
</head>
<body>
<div id="main-body">
<div id="contents">
<table>
<tr><th>Steps</th><th>Process</th></tr>
<tr><td>1</td><td>Figure out the <em>root element</em>.</td></tr>
<tr><td>2</td><td>Deal with the <span id="code">head</span> first,
as it's usually easy.</td></tr>
<tr><td>3</td><td>Work through the <span id="code">body</span>.
Just <em>take your time</em>.</td></tr>
</table>
</div>
<div id="closing">
This link is <em>not</em> active, but if it were, the answers
to this <a href="answers.html"><img src="exercise.gif" /></a> would
be there. But <em>do the exercise anyway!</em>
</div>
</div>
</body>
</html>
在 本文末的 GIF 文件 圖 2 中的 tricky-solution.gif 和 圖 3 中的 trickier-solution.gif 中將會找到這些練習的答案。不要偷看,先花些時間自動解答一下。這樣能幫助您理解組織樹時應用的規則有多麼嚴格,並真正幫助您掌握 HTML 及其樹結構。
屬性呢?
當您試圖弄清楚如何處理屬性時,是否遇到一些問題呢?前已說起,屬性確實具備本身的對象類型,但屬性確實不是顯示它的元素的子元素,嵌套元素和文本不在同一屬性 「級別」,您將注意到,清單 2 和 3 中練習的答案沒有顯示屬性。
屬性事實上存儲在瀏覽器使用的對象模型中,但它們有一些特殊狀況。每一個元素都有可用屬性的列表,且與子對象列表是分離的。因此 div 元素可能有一個包含屬性 「id」 和另外一個屬性 「class」 的列表。
記 住,元素的屬性必須具備唯一的名稱,也就是說,一個元素不能有兩個 「id」 或兩個 「class」 屬性。這使得列表易於維護和訪問。在下一篇文章將會看到,您能夠簡單調用諸如 getAttribute("id") 的方法來按名稱獲取屬性的值。還能夠用類似的方法調用來添加屬性或設置(重置)現有屬性的值。
值得指出的是,屬性名的唯一性使得該列表不 同於子對象列表。p 元素能夠有多個 em 元素,因此子對象列表能夠包含多個重複項。儘管子項列表和屬性列表的操做方式類似,但一個能夠包含重複項(對象的子項),而一個不能(元素對象的屬性)。 最後,只有元素具備屬性,因此文本對象沒有用於存儲屬性的附加列表。
凌亂的 HTML
在繼續以前,談到瀏覽器如何將標記轉換爲樹表示,還有一個主題值得探討,即瀏覽器如何處理不是格式良好的標記。格式良好 是 XML 普遍使用的一個術語,有兩個基本意思:
·每一個開始標記都有一個與之匹配的結束標記。因此每一個 <p> 在文檔中與 </p> 匹配,每一個 <div> 與 </div> 匹配,等等。
· 最裏面的開始標記與最裏面的結束標記相匹配,而後次裏面的開始標記與次裏面的結束標記相匹配,依此類推。因此 <b><i>bold and italics</b></i> 是不合法的,由於最裏面的開始標記 <i> 與最裏面的結束標記 <b> 匹配不當。要使之格式良好,要麼 切換開始標記順序,要麼 切換結束標記順序。(若是二者都切換,則仍會出現問題)。
深刻研究這兩條規則。這兩條規則不只簡化了文檔的組織,還消除了不定性。是否應先應用粗 體後應用斜體?或偏偏相反?若是以爲這種順序和不定性不是大問題,那麼請記住,CSS 容許規則覆蓋其餘規則,因此,例如,若是 b 元素中文本的字體不一樣於 i 元素中的字體,則格式的應用順序將變得很是重要。所以,HTML 的格式良好性有着舉足輕重的做用。
若是瀏 覽器收到了不是格式良好的文檔,它只會盡力而爲。獲得的樹結構在最好狀況下將是做者但願的原始頁面的近似,最壞狀況下將面目全非。若是您曾將頁面加載到瀏 覽器中後看到徹底出乎意料的結果,您可能在看到瀏覽器結果時會猜測您的結構應該如何,並沮喪地繼續工做。固然,搞定這個問題至關簡單:確保文檔是格式良好 的!若是不清楚如何編寫標準化的 HTML,請諮詢 參考資料 得到幫助。
DOM 簡介
到目前爲止,您已經知道瀏覽器將 Web 頁面轉換爲對象表示,可能您甚至會猜測,對象表示是 DOM 樹。DOM 表示 Document Object Model,是一個規範,可從 World Wide Web Consortium (W3C) 得到(您能夠參閱 參考資料 中的一些 DOM 相關連接)。
但更重要的是,DOM 定義了對象的類型和屬性,從而容許瀏覽器表示標記。(本系列下一篇文章將專門講述在 JavaScript 和 Ajax 代碼中使用 DOM 的規範。)
文檔對象
首先,須要訪問對象模型自己。這很是容易;要在運行於 Web 頁面上的任何 JavaScript 代碼中使用內置 document 變量,能夠編寫以下代碼:
var domTree = document;
固然,該代碼自己沒什麼用,但它演示了每一個 Web 瀏覽器使得 document 對象可用於 JavaScript 代碼,並演示了對象表示標記的完整樹(圖 1)。
每項都是一個節點
顯 然,document 對象很重要,但這只是開始。在進一步深刻以前,須要學習另外一個術語:節點。您已經知道標記的每一個部分都由一個對象表示,但它不僅是一個任意的對象,它是特 定類型的對象,一個 DOM 節點。更特定的類型,好比文本、元素和屬性,都繼承自這個基本的節點類型。因此能夠有文本節點、元素節點和屬性節點。
如 果已經有不少 JavaScript 編程經驗,那您可能已經在使用 DOM 代碼了。若是到目前爲止您一直在跟蹤本 Ajax 系列,那麼如今您必定 使用 DOM 代碼有一段時間了。例如,代碼行 var number = document.getElementById("phone").value; 使用 DOM 查找特定元素,而後檢索該元素的值(在本例中是一個表單字段)。因此即便您沒有意識到這一點,但您每次將 document 鍵入 JavaScript 代碼時都會使用 DOM。
詳細解釋已經學過的術語,DOM 樹是對象的樹,但更具體地說,它是節點 對象的樹。在 Ajax 應用程序中或任何其餘 JavaScript 中,能夠使用這些節點產生下列效果,好比移除元素及其內容,突出顯示特定文本,或添加新圖像元素。由於都發生在客戶端(運行在 Web 瀏覽器中的代碼),因此這些效果當即發生,而不與服務器通訊。最終結果一般是應用程序感受起來響應更快,由於當請求轉向服務器時以及解釋響應時,Web 頁面上的內容更改不會出現長時間的停頓。
在多數編程語言中,須要學習每種節點類型的實際對象名稱,學習可用的屬性,並弄清楚類型和強制轉換;但在 JavaScript 中這都不是必需的。您能夠只建立一個變量,併爲它分配您但願的對象(正如您已經看到的):
var domTree = document;
var phoneNumberElement = document.getElementById("phone");
var phoneNumber = phoneNumberElement.value;
沒有類型,JavaScript 根據須要建立變量併爲其分配正確的類型。結果,從 JavaScript 中使用 DOM 變得微不足道(未來有一篇文章會專門講述與 XML 相關的 DOM,那時將更加巧妙)。
結束語
在這裏,我要給您留一點懸念。顯然,這並不是是對 DOM 徹底詳盡的說明;事實上,本文不過是 DOM 的簡介。DOM 的內容要遠遠多於我今天介紹的這些!
本 系列的下一篇文章將擴展這些觀點,並深刻探討如何在 JavaScript 中使用 DOM 來更新 Web 頁面、快速更改 HTML 併爲您的用戶建立更交互的體驗。在後面專門講述在 Ajax 請求中使用 XML 的文章中,我將再次返回來討論 DOM。因此要熟悉 DOM,它是 Ajax 應用程序的一個主要部分。
此時,深刻了解 DOM 將十分簡單,好比詳細設計如何在 DOM 樹中移動、得到元素和文本的值、遍歷節點列表,等等,但這可能會讓您有這種印象,即 DOM 是關於代碼的,而事實上並不是如此。
在 閱讀下一篇文章以前,試着思考一下樹結構並用一些您本身的 HTML 實踐一下,以查看 Web 瀏覽器是如何將 HTML 轉換爲標記的樹視圖的。此外,思考一下 DOM 樹的組織,並用本文介紹的特殊狀況實踐一下:屬性、有元素混合在其中的文本、沒有 文本內容的元素(好比 img 元素)。
若是紮實掌握了這些概念,而後學習了 JavaScript 和 DOM 的語法(下一篇文章),則會使得響應更爲容易。
並且不要忘了,這裏有清單 2 和 3 的答案,其中還包含了示例代碼!
圖 2. 清單 2 的答案
圖 3. 清單 3 的答案