Ajax和XMLHttpRequest

这篇文章转发自我之前的一篇博客,是看过《你真的会使用XMLHttpRequest吗?》后结合自己之前对Ajax认识的总结。

Ajax:"Asynchronous JavaScript And XML"即异步JavaScript和XML,是一种创建交互式网页应用的网页开发技术,这项技术的核心是浏览器提供的XMLHttpRequest对象。

XMLHttpRequest的使用

1.设置request header

void setRequestHeader(headerName,headerValue)

  • setRequestHeader必须写在open()方法之后,send()方法之前

2. 获取response header

getAllResponseHeaders()

  • 获取全部可获取(有一些header是获取不到的)的header字段

getResponseHeader(headerName)

  • 获取指定header字段的值(只能获取部分header的值)

3. 指定xhr.response的数据类型

  • 通过设置xhr.response的类型,是告诉浏览器或者xhr如何处理响应数据(对响应数据按照什么样的格式去处理)

xhr.overrideMimeType()

  • 这个方法的作用就是重写response的MIME类型

  • 这个方法必须在send()方法之前进行调用

  • 可以通过这种方式,可以得到纯文本格式的图片内容,获取到数据后再使用相应的编码方法重新构建图片

var xhr = new XMLHtppRequest()
xhr.open('get','text.PHP',true)
xhr.overrideMimeType("text/xml; charset=utf-8")
xhr.send();

根据上面的例子,将response的MIME类型设置为了'text/xml',通过这样的方式,xhr会将响应当做text或者xml来处理,通过xhr.responsexhr.responseText可以获取到文本格式的相应数据,通过xhr.responseXML可以获取到XML格式(是一颗DOM树/DOM对象)的数据

var xhr = new XMLHtppRequest()
xhr.open('get',true)
xhr.overrideMimeType("text/plain; charset=utf-8")
xhr.send();

根据上面的例子,将response的MIME类型设置为了'text/plain',通过这样的方式,xhr会将相应当做text或者plain(纯文本)来处理,通过xhr.responsexhr.responseText可以获取到文本格式的相应数据,通过xhr.responseXML获取到的数据为null,即便数据是XML

xhr.responseType

responseType是XMLHttpRequest leve2中新增的属性,用来指定xhr.response的数据类型

var xhr = new XMLHttpRequest();
xhr.open('GET','/path/to/image.png',true);
//可以将`xhr.responseType`设置为`"blob"`也可以设置为`" arrayBuffer"`
//xhr.responseType = 'arrayBuffer';
xhr.responseType = 'blob';

xhr.onload = function(e) {
  if (this.status == 200) {
    // this.response就是Blob对象
    var blob = this.response;
    ...
  }
};

xhr.send();

4. 如何获取response数据

xhr提供了3个属性来获取请求返回的数据,分别是:xhr.response,xhr.responseText,xhr.responseXML

  • xhr.response:

    • 无论xhr.responseType的值是什么,只要请求完成,都可以从xhr.response中取到对应的值

    • 请求未完成时,xhr.response的值与xhr.responseType有关,xhr.responseType的值为""(默认值)或者text时,xhr.response的值为"",否则为null

  • xhr.responseText:

    • 只有当responseType设置为''或者text时,才可以使用这个属性获取相应内容

    • 请求未完成、失败时,该值为空字符串

  • xhr.responseXML:

    • 只有当 responseType 为''document时,此时才能调用xhr.responseXML,否则抛错

    • 请求未完成、失败时,该值为null

5. 如何追踪ajax请求的当前状态

  • xhr.readyState属性可以获取ajax的状态,每当xhr.readyState的值发生变化时,就会触发xhr.onreadystatechange事件,可以在这个事件中进行相应的操作

  • 为了保证跨浏览器兼容性,必须在调用open()方法之前指定onreadystatechange事件处理程序

xhr.onreadystatechange = function () {
    switch(xhr.readyState){
      case 1://OPENED
        //do something
            break;
      case 2://HEADERS_RECEIVED
        //do something
        break;
      case 3://LOADING
        //do something
        break;
      case 4://DONE
        //do something
         break;
    }

6. 设置请求的超时时间

  • XMLHttpRequest提供了timeout属性来允许设置请求的超时时间。

  • 从请求开始 算起,若超过 timeout 时间请求还没有结束(包括成功/失败),则会触发ontimeout事件,主动结束该请求。

  • 请求开始:xhr.onloadstart事件触发的时候,也就是你调用xhr.send()方法的时候

  • 请求结束:xhr.loadend事件触发的时候。

7. 发送同步请求

  • 由open()方法的第三个参数决定,当第三个参数async为true时,发送异步请求,为false时则为同步请求

  • 当xhr为同步请求时,有如下限制:

    • xhr.timeout必须为0

    • xhr.withCredentials必须为 false

    • xhr.responseType必须为""(即默认值)

  • 避免使用同步请求:因为我们无法设置请求超时时间(xhr.timeout为0,即不限时)。在不限制超时的情况下,有可能同步请求一直处于pending状态,服务端迟迟不返回响应,这样整个页面就会一直阻塞,无法响应用户的其他交互。

8. 如何获取上传、下载进度

我们可以通过onprogress事件来实时显示进度,默认情况下这个事件每50ms触发一次。需要注意的是,上传过程和下载过程触发的是不同对象的onprogress事件:

  • 上传触发的是xhr.upload对象的 onprogress事件

  • 下载触发的是xhr对象的onprogress事件

xhr.onprogress = updateProgress;
xhr.upload.onprogress = updateProgress;
function updateProgress(event) {
    if (event.lengthComputable) {
      var completedPercent = event.loaded / event.total;
    }
 }

xhr相关事件

每一XMLHttpRequest对象都有一个upload属性,而upload是一个XMLHttpRequestUpload对象.XMLHttpRequest和XMLHttpRequestUpload对象共同拥有上述(除onreadystatechange事件)事件。onreadystatechange是XMLHttpRequest对象独有的事件

1. 事件触发顺序

  1. 触发xhr.onreadystatechange(之后每次readyState变化时,都会触发一次)

  2. 触发xhr.onloadstart

上传阶段开始:

  1. 触发xhr.upload.onloadstart

  2. 触发xhr.upload.onprogress

  3. 触发xhr.upload.onload

  4. 触发xhr.upload.onloadend

上传结束,下载阶段开始:

  1. 触发xhr.onprogress

  2. 触发xhr.onload

  3. 触发xhr.onloadend

2. 发生abort/timeout/error异常的处理

  1. 一旦发生abort或timeout或error异常,先立即中止当前请求

  2. 将 readystate 置为4,并触发 xhr.onreadystatechange事件

  3. 如果上传阶段还没有结束,则依次触发以下事件:

    • xhr.upload.onprogress

    • xhr.upload.[onabort或ontimeout或onerror]

    • xhr.upload.onloadend

    • 注意不会触发onload事件

  4. 触发 xhr.onprogress事件

  5. 触发 xhr.[onabort或ontimeout或onerror]事件

  6. 触发xhr.onloadend 事件

  7. 注意不会触发onload事件

  • 注意: 这时候的xhr.readyState为4,xhr.status为0

3. 在哪个事件中注册成功回调

从上面介绍的事件中,可以知道若xhr请求成功,就会触发xhr.onreadystatechange和xhr.onload两个事件。由于xhr.onreadystatechange是每次xhr.readyState变化时都会触发,而不是xhr.readyState=4时才触发(例如发生abort、timeout、error异常,会先终止当前请求,将readyState设置为4,并触发onreadystatechange事件),因此建议在onload事件中注册成功回调

xhr.onload = function () {
    //如果请求成功
    if((xhr.status >= 200 && xhr.status < 300) || xhr.status == 304){
      //do successCallback
    }
  }

为什么要对xhr.status进行上述判断

xhr.status代表相应的HTTP状态
以2开头的状态码,代表请求已经成功被服务器接收、理解、并接受
状态代码304代表请求的资源并没有修改,可以直接使用浏览器中缓存的版本
其他以3开头的状态代码则表示需要客户端采取进一步的操作才能完成请求。

status和readyState

  • status是响应的HTTP状态,statusText是HTTP状态的说明

  • readyState是表示在请求/响应过程中的当前活动处于哪个阶段

参考文章

你真的会使用XMLHttpRequest吗?

20170615-Ajax和XMLHttpRequest的更多相关文章

  1. 关于h5中的fetch方法解读(小结)

    这篇文章主要介绍了关于h5中的fetch方法解读(小结),fetch身为H5中的一个新对象,他的诞生,是为了取代ajax的存在而出现,有兴趣的可以了解一下

  2. ios – 使用NSURLSession获取JSON数据

    我试图从谷歌距离api使用NSURLSession获取数据,但如下所示,当我打印响应和数据时,我得到的结果为NULL.可能是什么问题?

  3. ios – 错误域= com.alamofire.error.serialization.response代码= -1011“请求失败:禁止

    任何人都可以帮我解决以下错误–>在AFNetworking2.5中使用“删除”方法时出错解决方法我发现,如果我的手机时钟不同步……它不允许我更新…也许检查你的手机设置到正确的时间“自动区”,看看是否有效…

  4. iOS网页/原生应用Facebook登录弹出 – 失败?

    如果我重新启动app/web-app,用户将自动登录,并重定向到成功页面.我认为是导致问题的原因当您在Firefox/Chrome/Safari浏览器中运行网页时,Facebook登录对话框会弹出一个弹出窗口或另一个选项卡.我相信这是这个弹出页面的一个问题,以及当成功登录时Javascript如何与自身通信.window.close的东西没有返回的根页面…失败的解决方法由于应用程序挂在前面提到的URL上,我决定在shouldStartLoadWithRequest(…)中添加if语句以强制UIWebvie

  5. ios – Watchkit新会话不起作用

    我的手表扩展中有两个视图控制器.每当我打电话时我只得到第一个视图控制器的响应,并在第二个viewcontroller中得到错误WCSession在app和watch扩展中启动.任何建议?

  6. 使用Firebase iOS Swift将特定设备的通知推送到特定设备

    我非常感谢PushNotifications的帮助.我的应用聊天,用户可以直接向对方发送短信.但是如果没有PushNotifications,它就没有多大意义.它全部设置在Firebase上.如何将推送通知从特定设备发送到特定设备?

  7. ios – 保存从查询中获取的用户的属性(即不在currentUser上)

    我有兴趣根据currentUser执行的操作将属性保存到数据库中的用户.基于以下代码,我收到错误消息“除非已通过logIn或signUp验证用户,否则无法保存用户”我想知道是否有一个解决方法,我可以将属性保存到foundUser,而无需登录该用户.谢谢你的帮助!解决方法如果要更新当前不是登录用户的用户,则需要使用主密钥调用Parse.您可以从CloudCode执行此操作;并从您的iOS项目中调用它;

  8. 在iOS中使用NSJSONSerialization进行JSON解析

    解决方法首先在您的JSON响应字典中,在“RESPONSE”键下,您有一个数组而不是字典,该数组包含字典对象.所以要提取用户名和电子邮件ID,如下所示

  9. Xcode:Alamofire获得String响应

    我是IOS开发的新手,目前正在与Alamofire学习网络我正在尝试登录…每当凭证正确时,.PHP文件返回一个json,我可以通过以下代码从Alamofire获取json:现在……当凭证错误时,.PHP不会给json..而且它返回一个字符串..例如“wrong_password”或“userLocked”等等……如何通过Alamofire获得String响应?解决方法如果您希望JSON响应使用.responseJSON,如果您想要String响应,请使用.responseString.如果你想两者同时使用

  10. ios – 全局变量中的Appdelegate in swift

    我将一些数据从viewcontroller&从另一个视图控制器获取它.下面是应用程序委托的代码代码设置mainDic代码来获取字典.问题是我得到的输出没有.请让我正确.解决方法这是你的错误尝试将您的代码更改为此

随机推荐

  1. xe-ajax-mock 前端虚拟服务

    最新版本见Github,点击查看历史版本基于XEAjax扩展的Mock虚拟服务插件;对于前后端分离的开发模式,ajax+mock使前端不再依赖后端接口开发效率更高。CDN使用script方式安装,XEAjaxMock会定义为全局变量生产环境请使用xe-ajax-mock.min.js,更小的压缩版本,可以带来更快的速度体验。

  2. vue 使用 xe-ajax

    安装完成后自动挂载在vue实例this.$ajaxCDN安装使用script方式安装,VXEAjax会定义为全局变量生产环境请使用vxe-ajax.min.js,更小的压缩版本,可以带来更快的速度体验。cdnjs获取最新版本点击浏览已发布的所有npm包源码unpkg获取最新版本点击浏览已发布的所有npm包源码AMD安装require.js安装示例ES6Module安装通过Vue.use()来全局安装示例./Home.vue

  3. AJAX POST数据中文乱码解决

    前端使用encodeURI进行编码后台java.net.URLDecoder进行解码编解码工具

  4. Koa2框架利用CORS完成跨域ajax请求

    实现跨域ajax请求的方式有很多,其中一个是利用CORS,而这个方法关键是在服务器端进行配置。本文仅对能够完成正常跨域ajax响应的,最基本的配置进行说明。这样OPTIONS请求就能够通过了。至此为止,相当于仅仅完成了预检,还没发送真正的请求呢。

  5. form提交时,ajax上传文件并更新到&lt;input&gt;中的value字段

  6. ajax的cache作用

    filePath="+escape;},error:{alert;}});解决方案:1.加cache:false2.url加随机数正常代码:网上高人解读:cache的作用就是第一次请求完毕之后,如果再次去请求,可以直接从缓存里面读取而不是再到服务器端读取。

  7. 浅谈ajax上传文件属性contentType = false

    默认值为contentType="application/x-www-form-urlencoded".在默认情况下,内容编码类型满足大多数情况。在这里,我们主要谈谈contentType=false.在使用ajax上传文件时:在其中先封装了一个formData对象,然后使用post方法将文件传给服务器。说到这,我们发现在JQueryajax()方法中我们使contentType=false,这不是冲突了吗?这就是因为当我们在form标签中设置了enctype=“multipart/form-data”,

  8. 909422229_ajaxFileUpload上传文件

    ajaxFileUpload.js很多同名的,因为做出来一个很容易。我上github搜AjaxFileUpload出来很多类似js。ajaxFileUpload是一个异步上传文件的jQuery插件传一个不知道什么版本的上来,以后不用到处找了。语法:$.ajaxFileUploadoptions参数说明:1、url上传处理程序地址。2,fileElementId需要上传的文件域的ID,即的ID。3,secureuri是否启用安全提交,默认为false。4,dataType服务器返回的数据类型。6,error

  9. AJAX-Cache:一款好用的Ajax缓存插件

    原文链接AJAX-Cache是什么Ajax是前端开发必不可少的数据获取手段,在频繁的异步请求业务中,我们往往需要利用“缓存”提升界面响应速度,减少网络资源占用。AJAX-Cache是一款jQuery缓存插件,可以为$.ajax()方法扩展缓存功能。

  10. jsf – Ajax update/render在已渲染属性的组件上不起作用

    我试图ajax更新一个有条件渲染的组件。我可以确保#{user}实际上是可用的。这是怎么引起的,我该如何解决呢?必须始终在ajax可以重新呈现之前呈现组件。Ajax正在使用JavaScriptdocument.getElementById()来查找需要更新的组件。但是如果JSF没有将组件放在第一位,那么JavaScript找不到要更新的内容。解决方案是简单地引用总是渲染的父组件。

返回
顶部