熟悉web开发的程序员想必对Ajax也不会陌生。现在已经有很多js框架封装了ajax实现,例如JQuery的ajax函数,调用起来非常方便。当然本文不打算讲框架的使用,我们将从Ajax的javascript源码实现开始。
Ajax源码实现
var getXmlHttpRequest = function () {
if (window.XMLHttpRequest) {
//主流浏览器提供了XMLHttpRequest对象
return new XMLHttpRequest();
}
else if (window.ActiveXObject) {
//低版本的IE浏览器没有提供XMLHttpRequest对象
//所以必须使用IE浏览器的特定实现ActiveXObject
return new ActiveXObject("Microsoft.XMLHTTP");
}
};
var xhr = getXmlHttpRequest();
xhr.onreadystatechange = function () {
if (xhr.readyState === 4 && xhr.status === 200) {
//获取成功后执行操作
//数据在xhr.responseText
}
};
xhr.open("TYPE", "URL", true);
xhr.send("");
可以看到,xhr对象是通过onreadystatechange来监听Ajax的最终完成情况,这里也迎来了这次要重点讨论的主角:readyState和status。
什么是readyState
readyState是XMLHttpRequest对象的一个属性,用来标识当前XMLHttpRequest对象处于什么状态。
readyState总共有5个状态值,分别为0~4,每个值代表了不同的含义,如下表所示:
0 |
未初始化状态:此时,已经创建了一个XMLHttpRequest对象 |
1 |
准备发送状态:此时,已经调用了XMLHttpRequest对象的open方法,并且XMLHttpRequest对象已经准备好将一个请求发送到服务器端 |
2 |
已经发送状态:此时,已经通过send方法把一个请求发送到服务器端,但是还没有收到一个响应 |
3 |
正在接收状态:此时,已经接收到HTTP响应头部信息,但是消息体部分还没有完全接收到 |
4 |
完成响应状态:此时,已经完成了HTTP响应的接收 |
什么是status
status是XMLHttpRequest对象的一个属性,表示响应的HTTP状态码。
在HTTP1.1协议下,HTTP状态码总共可分为5大类,如下表所示:
1XX |
服务器收到请求,需要继续处理。例如101状态码,表示服务器将通知客户端使用更高版本的HTTP协议。 |
2XX |
请求成功。例如200状态码,表示请求所希望的响应头或数据体将随此响应返回。 |
3XX |
重定向。例如302状态码,表示临时重定向,请求将包含一个新的URL地址,客户端将对新的地址进行GET请求。 |
4XX |
客户端错误。例如404状态码,表示客户端请求的资源不存在。 |
5XX |
服务器错误。例如500状态码,表示服务器遇到了一个未曾预料的情况,导致了它无法完成响应,一般来说,这个问题会在程序代码出错时出现。 |
抛出问题
为什么onreadystatechange的函数实现要同时判断readyState和status呢?
我们知道 readyState === 4 已经表示了请求响应成功了,为什么还要后续的status呢?带着问题,我们开始来做一些试验吧。
只使用readyState判断
javascript端的实现代码如下:
var getXmlHttpRequest = function () {
if (window.XMLHttpRequest) {
return new XMLHttpRequest();
}
else if (window.ActiveXObject) {
return new ActiveXObject("Microsoft.XMLHTTP");
}
};
var xhr = getXmlHttpRequest();
xhr.onreadystatechange = function () {
if (xhr.readyState === 4) {
alert(xhr.responseText);
}
};
xhr.open("GET", "/data.aspx", true);
xhr.send("");
这个演示DEMO使用ASP.NET Webform框架,现在我们在后台data.aspx做一些手脚,不如让它报错试试!C#代码如下:
public partial class data : System.Web.UI.Page
{
protected void Page_Load(object sender, EventArgs e)
{
throw new Exception("Error");
}
}
运行javascript代码,提示窗口出现了如下:
服务响应出错了,但还是返回了信息,这并不是我们想要的结果。打开Fiddler监控,可以看到data.aspx返回的是500响应,但由于只使用readystate做判断,它不理会放回的结果是500还是200,只要响应成功返回了,就执行接下来的javascript代码,结果将造成各种不可预料的错误。所以只使用readyState判断是行不通的。
换另外一个角度想,状态码返回200就表示这次响应是成功的了,那么是不是可以不使用readyState,单独只使用status做判断呢?好,带着问题,继续来做试验吧。
只使用status判断
javascript端的代码实现如下:
var getXmlHttpRequest = function () {
if (window.XMLHttpRequest) {
return new XMLHttpRequest();
}
else if (window.ActiveXObject) {
return new ActiveXObject("Microsoft.XMLHTTP");
}
};
var xhr = getXmlHttpRequest();
xhr.onreadystatechange = function () {
if (xhr.status === 200) {
alert("readyState=" + xhr.readyState + xhr.responseText);
}
};
xhr.open("GET", "/data.aspx", true);
xhr.send("");
这次将在data.aspx后台做处理,让它只返回一个字符串,实现如下:
public partial class data : System.Web.UI.Page
{
protected void Page_Load(object sender, EventArgs e)
{
Response.Write("Test");
Response.End();
}
}
一切都是那么地自然,是不是只要弹出一个写着一行"readyState=4Test"的字符串的提示框,就表示结果成立了?把它跑起来了吧,结果已经就离我们不远了!
事实上,结果却不像预期那样。响应码确实是返回了200,但是总共弹出了3次窗口!第一次是“readyState=2”的窗口,第二次是“readyState=3Test”的窗口,第三次是“readyState=4Test”的窗口。由此,可见onreadystatechange函数的执行不是只在readyState变为4的时候触发的,而是readyState的每次变化都会触发,所以就出现了前面说的那种情况。可见,单独使用status判断也是行不通的。
进一步思考
由上面的试验,我们可以知道判断的时候readyState和status缺一不可。那么readyState和status的先后判断顺序会不会有影响呢?我们可以将status调到前面先判断,代码如 xhr.status === 200 && xhr.readyState === 4。
事实上,这对于最终的结果是没有影响的,但是中间的性能就不同了。由上一个试验我们知道,readyState的每次变化都会触发onreadystatechange函数,假如先判断status,那么每次都会多判断一次status的状态。虽然性能上影响甚微,不过我们还是应该抱着追求极致代码的想法,把readyState的判断放在前面。
参考资料: 锋利的JQuery, http状态码维基百
相关推荐
在探讨jQuery AJAX技术时,两个重要的属性是readyState和status。它们在异步请求中扮演了重要角色,但它们各自的含义和使用场景有所不同。通过理解两者的区别与联系,可以更好地管理AJAX请求的生命周期,以及响应...
在前几篇分析了jquery的ajax异步和同步,以及异常的一些处理,感觉还没有把ajax的readyState和status说清楚.今天就来说说ajax状态的那点事。 jquery ajax函数源代码是这样的: var getXmlHttpRequest = function ()...
了解这些状态对于理解和调试Ajax请求至关重要。 1. **状态0 (Uninitialized)**: 这是初始状态,XMLHttpRequest对象已经被创建,但还没有调用`open()`方法来初始化请求。在这个状态下,请求还没有开始,对象处于待命...
在Ajax技术中,readyState和status是两个至关重要的属性,它们分别反映了Ajax请求的不同阶段和服务器响应的状态。本文将深入探讨这两个属性的概念、用途以及它们之间的区别。 首先,`readyState`是XMLHttpRequest...
- **jQuery 和 Axios**:学习Ajax时,可以了解jQuery的$.ajax()方法和现代JavaScript库Axios,它们都提供了更方便的接口来处理Ajax请求。 在项目中,Ajax常用于实现动态加载、表单提交、实时通讯等功能。随着前端...
开发者可以注册这些事件的回调函数,以便在合适的时候执行相应的操作,例如检查readyState和status属性,确定请求的状态和成功与否。 3. **服务器响应**:服务器接收到请求后,处理请求并返回数据,通常是XML、JSON...
它提供了open()、send()等方法以及各种属性,如readyState和status,用于控制和监测请求过程。此外,还可以使用不同的HTTP请求类型,如GET、POST、PUT、DELETE等,以适应不同类型的交互需求。 在Ajax应用中,理解...
以下是关于AJAX `readyState`五种状态的详细解释: 1. **未初始化 (0)**: 这是`XMLHttpRequest`对象创建后的初始状态。在这个状态下,`open()`方法尚未被调用,请求还未开始。此时,开发者通常会设置请求的URL、...
如果`readyState`属性为4(表示请求已完成),且`status`属性为200(表示成功),则可以访问响应数据。 - **处理响应**:根据返回的数据类型(可能是XML、JSON或其他格式),解析并更新网页内容。 ### 3. **使用...
当服务器响应时,会触发onreadystatechange事件,我们可以通过检查readyState和status属性来判断请求的状态和成功与否。 在接收到服务器响应后,我们通常会使用JavaScript处理返回的数据,这可能是XML、JSON或其他...
- 注册 `onreadystatechange` 事件处理函数,检查 `readyState` 和 `status` 状态,确保请求已完成且成功。 4. **接收回应** 服务器的回应通过 `onreadystatechange` 事件处理函数来处理。当 `readyState` 为 4 ...
在处理Ajax请求时,通常会在`.onreadystatechange`事件中检查`readyState`和`status`。例如: ```javascript var xhr = new XMLHttpRequest(); xhr.onreadystatechange = function() { if (xhr.readyState === 4) ...
Ajax 简介和 Ajax API Ajax 简介 Ajax 是“Asynchronous JavaScript and XML”(异步 JavaScript 和 XML)的缩写词,但事实上,Ajax 并非缩写词,而是由 Jesse James Gaiett 创造的名词。Ajax 是一种创建交互式...
当服务器响应时,通过onreadystatechange事件处理程序监听状态变化,判断readyState和status属性,确保数据已成功接收。 二、Div窗口 Div(Division)是HTML中的一个块级元素,用于分隔网页内容。我们可以用CSS来...
3. **事件监听与处理**:学习如何监听请求状态变化,如onreadystatechange事件,以及如何判断请求是否成功(readyState和status属性)。 4. **JSON数据格式**:在Ajax中,XML已较少使用,更多的是JSON格式。了解...
下面我们将详细讨论Ajax的基本使用和XML的解析方式。 1. Ajax基本使用: - 创建XMLHttpRequest对象:在JavaScript中,首先需要创建一个XMLHttpRequest对象,这是Ajax的核心。 - 配置请求:设置HTTP请求的类型...
if (xhr.readyState === 4 && xhr.status === 200) { var response = JSON.parse(xhr.responseText); // 处理验证结果 } }; xhr.send(JSON.stringify({username: username, password: password})); } ``` -...