`
icybamboo
  • 浏览: 39830 次
  • 性别: Icon_minigender_1
  • 来自: 杭州
社区版块
存档分类
最新评论

XMLHTTP.readyState的五种状态(转)

阅读更多

对于很多 Web 开发人员来说,只需要生成简单的请求并接收简单的响应即可;但是对于希望掌握 Ajax 的开发人员来说,必须要全面理解 HTTP 状态代码、就绪状态和 XMLHttpRequest 对象。在本文中,Brett McLaughlin 将向您介绍各种状态代码,并展示浏览器如何对其进行处理,本文还给出了在 Ajax 中使用的比较少见的 HTTP 请求。

在本系列的 上篇文章 中,我们将详细介绍 XMLHttpRequest 对象,它是 Ajax 应用程序的中心,负责处理服务器端应用程序和脚本的请求,并处理从服务器端组件返回的数据。由于所有的 Ajax 应用程序都要使用 XMLHttpRequest 对象,因此您可能会希望熟悉这个对象,从而能够让 Ajax 执行得更好。
在本文中,我将在上一篇文章的基础上重点介绍这个请求对象的 3 个关键部分的内容:

HTTP 就绪状态
HTTP 状态代码
可以生成的请求类型
这 三部分内容都是在构造一个请求时所要考虑的因素;但是介绍这些主题的内容太少了。然而,如果您不仅仅是想了解 Ajax 编程的常识,而是希望了解更多内容,就需要熟悉就绪状态、状态代码和请求本身的内容。当应用程序出现问题时 —— 这种问题总是存在 —— 那么如果能够正确理解就绪状态、如何生成一个 HEAD 请求或者 400 的状态代码的确切含义,就可以在 5 分钟内调试出问题,而不是在各种挫折和困惑中度过 5 个小时。
XMLHttpRequest 或 XMLHttp:换名玫瑰
Microsoft™ 和 Internet Explorer 使用了一个名为 XMLHttp 的对象,而不是 XMLHttpRequest 对象,而 Mozilla、Opera、Safari 和 大部分非 Microsoft 浏览器都使用的是后者。为了简单性起见,我将这两个对象都简单地称为 XMLHttpRequest。这既符合 我们在 Web 上看到的情况,又符合 Microsoft 在 Internet Explorer 7.0 中使用 XMLHttpRequest 作为请求对象的意图。(有关这个问题的更多内容,请参见 第 2 部分。)
下 面让我们首先来看一下 HTTP 就绪状态。
深入了解 HTTP 就绪状态
您应该还记得在上一篇文章中 XMLHttpRequest 对象有一个名为 readyState 的属性。这个属性确保服务器已经完成了一个请求,通常会使用一个回调函数从服务器中读出数据来更新 Web 表单或页面的内容。清单 1 给出了一个简单的例子(这也是本系列的上一篇文章中的一个例子 —— 请参见 参考资料)。

清单 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

当 0 等于 4 时
在多个 JavaScript 函数都使用相同的请求对象时,您需要检查就绪状态 0 来确保这个请求对象没有正在使用,这种机制会产生问题。由于 readyState == 4 表示一个已完成的请求,因此您经常会发现那些目前没在使用的处于就绪状态的请求对象仍然被设置成了 4 —— 这是因为从服务器返回来的数据已经使用过了,但是从它们被设置为就绪状态之后就没有进行任何变化。有一个函数 abort() 会重新设置请求对象,但是这个函数却不是真正为了这个目的而使用的。如果您 必须 使用多个函数,最好是为每个函数都创建并使用一个函数,而不是在多个函数之间共享相同的对象。
显然,这并不能为您带来多少好处;需要确保 尚未 调用 open() 函数的情况很少。在大部分 Ajax 编程的真实情况中,这种就绪状态的唯一用法就是使用相同的 XMLHttpRequest 对象在多个函数之间生成多个请求。在这种(不常见的)情况中,您可能会在生成新请求之前希望确保请求对象是处于未初始化状态(readyState == 0)。这实际 上是要确保另外一个函数没有同时使用这个对象。
查看正在处理的请求的就绪状态
除了 0 就绪状态之外,请求对象还需要依次经历典型的请求和响应的其他几种就绪状态,最后才以就绪状态 4 的形式结束。这就是为什么您在大部分回调函数中都可以看到 if (request.readyState == 4) 这行代码;它确保服务器已经完成对请求的处理,现在可以安全地更新 Web 页面或根据从服务器返回来的数据来进行操作了。
要查看这种状态发生的 过程非常简单。如果就绪状态为 4,我们不仅要运行回调函数中的代码,而且还要在每次调用回调函数时都输出就绪状态。 清单 3 给出了一个实现这种功能的例子。

清单 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)。由于这个原因,我将这段代码留作练习,而没有在本文中包括进来。
现 在应该来看一下状态代码了。
分享到:
评论

相关推荐

    xmlhttp.rar_xmlhttp

    3. **处理响应**:通过`ReadyState`属性监控请求状态,当`ReadyState`为4(表示请求完成)且`Status`为200(表示成功)时,可以通过`ResponseText`或`ResponseXML`属性获取服务器响应的数据,前者为文本形式,后者为...

    XmlHttp.rar_xmlhttp

    6. **readyState** 属性:表示请求的当前状态,从0到4,0表示请求未初始化,4表示请求已完成且响应已就绪。 7. **onreadystatechange** 事件:当readyState属性改变时触发,常用于检查请求是否完成并处理响应数据。...

    VB使用Microsoft.XMLHTTP判断程序是否联入互联网

    - 使用 `Do...Loop` 循环检查 `xmlHttp.ReadyState` 的状态,直到请求完成或超时。 - `DoEvents` 语句使得控制权短暂地交给操作系统,以便可以处理其他事件。 6. **响应处理**: - 如果 `xmlHttp.ResponseText` ...

    XMLHTTP.rar_xmlhttp

    - `readyState`:表示请求的状态,共有五种状态,0表示未初始化,1表示已创建,2表示已发送,3表示接收中,4表示完成。 - `status`:返回HTTP请求的状态码,如200表示成功,404表示未找到资源,500表示服务器内部...

    WEB开发 之 AJAX - onreadystatechange 事件.docx

    `readyState`属性有五个不同的状态值,分别代表了请求的不同阶段: 1. `0 (UNSENT)`:XMLHttpRequest对象已经创建,但还没有调用`open()`方法。 2. `1 (OPENED)`:`open()`方法已经被调用,但请求还没有发送。 3. `...

    xmlhttp.rar_刷新 数据库

    "xmlhttp.rar_刷新 数据库"这个压缩包文件很可能包含了一个关于如何使用XHR对象从数据库获取数据的示例。 在Web应用中,用户通常希望页面能够快速响应,而无需等待整个页面重新加载。这就是XHR发挥作用的地方。...

    关于HTTP及XMLHTTP状态代码一览.txt

    这些状态码分为五类:1xx(信息性响应)、2xx(成功响应)、3xx(重定向响应)、4xx(客户端错误)以及5xx(服务器错误)。接下来将详细介绍这些状态码。 ##### 1xx 信息性响应 - **100 Continue**:表示客户端...

    xmlhttp.rar_xmlhttp_xmlhttprequest

    `readyState`属性表示请求的五个阶段,0-4分别代表未初始化、已连接、头部已接收、响应体加载中、完成。通常关注`readyState == 4`,即请求已完成。 6. **获取响应信息**:`status`属性返回HTTP状态码,如200表示...

    xmlhttp.chm

    5. `readyState`属性:表示请求的当前状态,从0到4,4代表完成。 6. `status`属性:返回HTTP状态码,如200表示成功,404表示未找到资源。 7. `responseText`和`responseXML`属性:分别获取响应的文本和XML格式数据。...

    ajax项目实例

    `xmlhttp.readyState`表示请求状态,4表示完成;`xmlhttp.status`为HTTP状态码,200表示成功。 三、处理响应 响应数据通常以文本或XML形式返回。通过`xmlhttp.responseText`获取文本,`xmlhttp.responseXML`获取XML...

    XMLHttpRequest

    if (xmlhttp.readyState == 4) { myButton.disabled = false; alert("Result=" + xmlhttp.responseText); } } ``` 2. **`readyState`**: - **作用**:表示请求/响应过程中的当前活动阶段。 - **取值**: ...

    WEB开发 之 AJAX - 向服务器发送请求.docx

    if (xmlhttp.readyState == 4 && xmlhttp.status == 200) { document.getElementById("myDiv").innerHTML = xmlhttp.responseText; } } ``` 使用AJAX技术可以实现客户端和服务器之间的异步通信,提高Web应用程序...

    用法JavaScript实现ajax的实例代码_.docx

    AJAX(Asynchronous JavaScript and XML)是一种用于创建交互式网页应用的技术,它使得网页可以在无需刷新整个页面的情况下,仅更新部分数据。这种技术极大地提升了用户体验,因为它减少了等待时间和页面跳转,使得...

    AJAX技术操作的三部曲

    当请求完成,即 `xmlHttp.readyState` 变为4(表示请求已完成),并且 `xmlHttp.status` 是200(表示服务器成功返回)时,我们可以读取并处理服务器返回的数据。在示例中,`handleStateChange()` 函数检查响应文本...

    h实验八 javascript实验的网页加载进度条(ajax) - 副本.doc

    `XmlHttp.readyState`属性表示请求的状态,`XmlHttp.onreadystatechange`事件处理函数会在状态改变时被调用。 3. **AJAX发送请求**:`Ajax.Send`方法负责向服务器发送请求。`XmlHttp.open("GET", url, true)`开启一...

    原生JavaScript实现Ajax的方法_.docx

    1. **状态检查**:在处理响应时,需要检查`readyState`和`status`属性来确保请求已完成且响应成功。 2. **编码问题**:在传递参数时,最好使用`encodeURIComponent`来处理参数,以避免乱码问题。 3. **跨域问题**:...

    神奇的_ajax框架

    if (xmlHttp.readyState == 4 && xmlHttp.status == 200) { document.getElementById('result').innerHTML = xmlHttp.responseText; } }; ``` - `readyState`属性表示请求/响应过程中的不同状态: - **0**:...

Global site tag (gtag.js) - Google Analytics