fetch和XMLHttpRequest

如果看网上的fetch教程,会首先对比XMLHttpRequest和fetch的优劣,然后引出一堆看了很快会忘记的内容(本人记性不好)。因此,我写一篇关于fetch的文章,为了自己看着方便,毕竟工作中用到的也就是一些很基础的点而已。

fetch,说白了,就是XMLHttpRequest的一种替代方案。如果有人问你,除了Ajax获取后台数据之外,还有没有其他的替代方案?

这是你就可以回答,除了XMLHttpRequest对象来获取后台的数据之外,还可以使用一种更优的解决方案fetch

如何获取fetch

到现在为止,fetch的支持性还不是很好,但是在谷歌浏览器中已经支持了fetch。fetch挂在在BOM中,可以直接在谷歌浏览器中使用。

查看fetch的支持情况:fetch的支持情况

当然,如果不支持fetch也没有问题,可以使用第三方的ployfill来实现只会fetch:whatwg-fetch

fetch的helloworld

下面我们来写第一个fetch获取后端数据的例子:

  1. // 通过fetch获取百度的错误提示页面
  2. fetch('https://www.baidu.com/search/error.html') // 返回一个Promise对象
  3. .then((res)=>{
  4. return res.text() // res.text()是一个Promise对象
  5. })
  6. .then((res)=>{
  7. console.log(res) // res是最终的结果
  8. })

是不是很简单?可能难的地方就是Promise的写法,这个可以看阮一峰老师的ES6教程来学习。

说明一点,下面演示的GET请求或POST请求,都是采用百度中查询到的一些接口,可能传递的有些参数这个接口并不会解析,但不会影响这个接口的使用。

GET请求

GET请求初步

完成了helloworld,这个时候就要来认识一下GET请求如何处理了。

上面的helloworld中这是使用了第一个参数,其实fetch还可以提供第二个参数,就是用来传递一些初始化的信息。

这里如果要特别指明是GET请求,就要写成下面的形式:

  1. // 通过fetch获取百度的错误提示页面
  2. fetch('https://www.baidu.com/search/error.html', {
  3. method: 'GET'
  4. })
  5. .then((res)=>{
  6. return res.text()
  7. })
  8. .then((res)=>{
  9. console.log(res)
  10. })

GET请求的参数传递

GET请求中如果需要传递参数怎么办?这个时候,只能把参数写在URL上来进行传递了。

  1. // 通过fetch获取百度的错误提示页面
  2. fetch('https://www.baidu.com/search/error.html?a=1&b=2', { // 在URL中写上传递的参数
  3. method: 'GET'
  4. })
  5. .then((res)=>{
  6. return res.text()
  7. })
  8. .then((res)=>{
  9. console.log(res)
  10. })

POST请求

POST请求初步

与GET请求类似,POST请求的指定也是在fetch的第二个参数中:

  1. // 通过fetch获取百度的错误提示页面
  2. fetch('https://www.baidu.com/search/error.html', {
  3. method: 'POST' // 指定是POST请求
  4. })
  5. .then((res)=>{
  6. return res.text()
  7. })
  8. .then((res)=>{
  9. console.log(res)
  10. })

POST请求参数的传递

众所周知,POST请求的参数,一定不能放在URL中,这样做的目的是防止信息泄露。

  1. // 通过fetch获取百度的错误提示页面
  2. fetch('https://www.baidu.com/search/error.html', {
  3. method: 'POST',
  4. body: new URLSearchParams([["foo", 1],["bar", 2]]).toString() // 这里是请求对象
  5. })
  6. .then((res)=>{
  7. return res.text()
  8. })
  9. .then((res)=>{
  10. console.log(res)
  11. })

其实除了对象URLSearchParams外,还有几个其他的对象,可以参照:常用的几个对象来学习使用。

设置请求的头信息

在POST提交的过程中,一般是表单提交,可是,经过查询,发现默认的提交方式是:Content-Type:text/plain;charset=UTF-8,这个显然是不合理的。下面咱们学习一下,指定头信息:

  1. // 通过fetch获取百度的错误提示页面
  2. fetch('https://www.baidu.com/search/error.html', {
  3. method: 'POST',
  4. headers: new Headers({
  5. 'Content-Type': 'application/x-www-form-urlencoded' // 指定提交方式为表单提交
  6. }),
  7. body: new URLSearchParams([["foo", 1],["bar", 2]]).toString()
  8. })
  9. .then((res)=>{
  10. return res.text()
  11. })
  12. .then((res)=>{
  13. console.log(res)
  14. })

这个时候,在谷歌浏览器的Network中查询,会发现,请求方式已经变成了content-type:application/x-www-form-urlencoded

通过接口得到JSON数据

上面所有的例子中都是返回一个文本,那么除了文本,有没有其他的数据类型呢?肯定是有的,具体查询地址:Body的类型

由于最常用的是JSON数据,那么下面就简单演示一下获取JSON数据的方式:

  1. // 通过fetch获取百度的错误提示页面
  2. fetch('https://www.baidu.com/rec?platform=wise&ms=1&rset=rcmd&word=123&qid=11327900426705455986&rq=123&from=844b&baiduid=A1D0B88941B30028C375C79CE5AC2E5E%3AFG%3D1&tn=&clientWidth=375&t=1506826017369&r=8255', { // 在URL中写上传递的参数
  3. method: 'GET',
  4. headers: new Headers({
  5. 'Accept': 'application/json' // 通过头指定,获取的数据类型是JSON
  6. })
  7. })
  8. .then((res)=>{
  9. return res.json() // 返回一个Promise,可以解析成JSON
  10. })
  11. .then((res)=>{
  12. console.log(res) // 获取JSON数据
  13. })

强制带Cookie

默认情况下, fetch 不会从服务端发送或接收任何 cookies, 如果站点依赖于维护一个用户会话,则导致未经认证的请求(要发送 cookies,必须发送凭据头).

  1. // 通过fetch获取百度的错误提示页面
  2. fetch('https://www.baidu.com/search/error.html', {
  3. method: 'GET',
  4. credentials: 'include' // 强制加入凭据头
  5. })
  6. .then((res)=>{
  7. return res.text()
  8. })
  9. .then((res)=>{
  10. console.log(res)
  11. })

简单封装一下fetch

最后了,介绍了一大堆内容,有没有发现,在GET和POST传递参数的方式不同呢?下面咱们就来封装一个简单的fetch,来实现GET请求和POST请求参数的统一。

  1. /**
  2. * 将对象转成 a=1&b=2的形式
  3. * @param obj 对象
  4. */
  5. function obj2String(obj, arr = [], idx = 0) {
  6. for (let item in obj) {
  7. arr[idx++] = [item, obj[item]]
  8. }
  9. return new URLSearchParams(arr).toString()
  10. }
  11. /**
  12. * 真正的请求
  13. * @param url 请求地址
  14. * @param options 请求参数
  15. * @param method 请求方式
  16. */
  17. function commonFetcdh(url, options, method = 'GET') {
  18. const searchStr = obj2String(options)
  19. let initObj = {}
  20. if (method === 'GET') { // 如果是GET请求,拼接url
  21. url += '?' + searchStr
  22. initObj = {
  23. method: method,
  24. credentials: 'include'
  25. }
  26. } else {
  27. initObj = {
  28. method: method,
  29. credentials: 'include',
  30. headers: new Headers({
  31. 'Accept': 'application/json',
  32. 'Content-Type': 'application/x-www-form-urlencoded'
  33. }),
  34. body: searchStr
  35. }
  36. }
  37. fetch(url, initObj).then((res) => {
  38. return res.json()
  39. }).then((res) => {
  40. return res
  41. })
  42. }
  43. /**
  44. * GET请求
  45. * @param url 请求地址
  46. * @param options 请求参数
  47. */
  48. function GET(url, options) {
  49. return commonFetcdh(url, options, 'GET')
  50. }
  51. /**
  52. * POST请求
  53. * @param url 请求地址
  54. * @param options 请求参数
  55. */
  56. function POST(url, options) {
  57. return commonFetcdh(url, options, 'POST')
  58. }
  1. GET('https://www.baidu.com/search/error.html', {a:1,b:2})
  2. POST('https://www.baidu.com/search/error.html', {a:1,b:2})

fetch,终于认识你的更多相关文章

  1. 终于开始用github了

    一直以来,github的大名就如雷贯耳.虽然我半年多前就从了解到了这个神奇的网站,而且趁着当时的一时兴趣注册了账户,但是对于那时候的我来说这个网站还是太复杂了点,毕竟半年前的我还没有开始写代码啊,所以 ...

  2. [转] 传统 Ajax 已死,Fetch 永生

    原谅我做一次标题党,Ajax 不会死,传统 Ajax 指的是 XMLHttpRequest(XHR),未来现在已被 Fetch 替代. 最近把阿里一个千万级 PV 的数据产品全部由 jQuery 的  ...

  3. 记一次数据库调优过程(IIS发过来SQLSERVER 的FETCH API_CURSOR语句是神马?)

    记一次数据库调优过程(IIS发过来SQLSERVER 的FETCH API_CURSOR语句是神马?) 前几天帮客户优化一个数据库,那个数据库的大小是6G 这麽小的数据库按道理不会有太大的性能问题的, ...

  4. nVIDIA SDK White Paper ----Vertex Texture Fetch Water

    http://blog.csdn.net/soilwork/article/details/713842 nVIDIA SDK White Paper ----Vertex Texture Fetch ...

  5. 使用国内镜像通过pip安装python的一些包 Cannot fetch index base URL http://pypi.python.org/simple/

    原文地址:http://www.xuebuyuan.com/1157602.html 学习flask,安装virtualenv环境,这些带都ok,但是一安装包总是出错无法安装, 比如这样超时的问题: ...

  6. 传统 Ajax 已死,Fetch 永生

    原谅我做一次标题党,Ajax 不会死,传统 Ajax 指的是 XMLHttpRequest(XHR),未来现在已被 Fetch 替代. 最近把阿里一个千万级 PV 的数据产品全部由 jQuery 的  ...

  7. Fetch和ajax的比较和区别

    传统 Ajax 已死,Fetch 永生   Ajax 不会死,传统 Ajax 指的是 XMLHttpRequest(XHR),未来现在已被 Fetch 替代. 最近把阿里一个千万级 PV 的数据产品全 ...

  8. weex stream 之fetch的get、post获取Json数据

    无论何种平台,网络数据的获取都是十分重要的,最近学习weex,不可避免的要学习weex的数据请求方法了.网址 个人感觉,weex stream相较于其他平台,还算比较简单了,但是由于文档以及官方代码中 ...

  9. [2013.7.5新鲜出炉] Ubuntu12.04下载Android4.0.1源码全过程----------------折腾两天,终于下好,附若干问题解决

    本文转至 http://blog.csdn.net/yanzi1225627/article/details/9255457 下载源码这一步折腾了我整整两天,期间遇到很多问题,哎,记录于此,希望日后再 ...

随机推荐

  1. npm cnpm yarn 安装

    安装node.js,其中已经集成了npm,可以将npm切换到国内镜像 $ npm config set registry https://registry.npm.taobao.org -- 配置后可 ...

  2. js和C# 编码 解码

    C#中对URL编码的方法... 编码:Server.UrlEncode(string) 解码:Server.UrlDecode(string) HttpUtility.UrlEncode(string ...

  3. MM32F0020 UART1中断接收和UART1中断发送

    目录: 1.MM32F0020简介 2.初始化MM32F0020 UART1和NVIC中断 3.编写MM32F0020 UART1使能中断发送函数 4.编写MM32F0020 UART1中断接收和中断 ...

  4. JZ-056-删除链表中重复的结点

    删除链表中重复的结点 题目描述 在一个排序的链表中,存在重复的结点,请删除该链表中重复的结点,重复的结点不保留,返回链表头指针. 例如,链表1->2->3->3->4-> ...

  5. jq 简易购物车功能实现

    <!DOCTYPE html><html> <head> <meta charset="utf-8"> <title>& ...

  6. 微信小程序授权登录将open_id传至后台并入库

    要求能把用户昵称.头像以及open_id写入数据库,服务端保持用户登录状态 wxml: <block wx:else> <button type="primary" ...

  7. nginx 配置 https,及加载配置文件夹

    首先需要去申请一个域名签名证书,在腾讯云,阿里云都有免费版,然后下载下来按如下配置,请根据自己路径更改 server { listen 80; server_name xxx.xxx.cn; root ...

  8. webug 4.0 打靶笔记-01

    webug 4.0 打靶笔记 1. 显错注入 1.1 访问靶场 1.2 判断注入点 查找一切有参数传入的地方进行测试,注意到有get传参?id=1 猜测后台php中sql语句模板可能为如下几种情况 $ ...

  9. [链表]LeetCode 25 K组一个翻转链表

    LeetCode 25 k组一个翻转链表 TITLE 示例 1: 输入:head = [1,2,3,4,5], k = 2 输出:[2,1,4,3,5] 示例 2: 输入:head = [1,2,3, ...

  10. Linux卸载源码编译安装的软件

    使用auto-apt 和 checkinstall,具体命令如下 #安装auto-apt和checkinstall apt install auto-apt checkinstall #在源码目录中 ...