引言
前端是一个快速发展的领域,而在前端的技术栈当中,前端请求又是最见的一个领域,通过请求接口数据,才能将一个静态的页面动态化。本文将以前端发展的时间轴来逐一分析前端请求的技术演变及其优劣。
XMLHttpRequest
XMLHttpRequest是最早出现的与服务器交换数据的方案,有了XMLHttpRequest,开发者终于可以在不重新加载页面的情况下更新网页,可以在页面加载后请求接受以及发送数据。而所有的浏览器均支持获取XMLHttpRequest对象
var xhr = new XMLHttpRequest(); //获取XMLHttpRequest对象
由于浏览器的兼容性,各个浏览器对于创建XMLHttpRequest的方法不同,以下是兼容方法:
var xhr;
if (window.XMLHttpRequest) { // Mozilla, Safari...
xhr = new XMLHttpRequest();
} else if (window.ActiveXObject) { // IE
try {
xhr = new ActiveXObject('Msxml2.XMLHTTP');
} catch (e) {
try {
xhr = new ActiveXObject('Microsoft.XMLHTTP'); //IE5,6
} catch (e) {}
}
}
使用XMLHttpRequest发起一个get请求
// get请求
xhr.open("GET","test1.txt",true);
xhr.send();
完整的post请求代码如下:
var xhr;
if (window.XMLHttpRequest) { // Mozilla, Safari...
xhr = new XMLHttpRequest();
} else if (window.ActiveXObject) { // IE
try {
xhr = new ActiveXObject('Msxml2.XMLHTTP');
} catch (e) {
try {
xhr = new ActiveXObject('Microsoft.XMLHTTP');
} catch (e) {}
}
}
if (xhr) {
xhr.onreadystatechange = onReadyStateChange;
xhr.open('POST', '/api', true);
// 设置 Content-Type 为 application/x-www-form-urlencoded
// 以表单的形式传递数据
xhr.setRequestHeader('Content-Type', 'application/x-www-form-urlencoded');
xhr.send('username=admin&password=root');
}
// onreadystatechange 方法
function onReadyStateChange() {
// 该函数会被调用四次
if (xhr.readyState === 4 &&xhr.status === 200) {
console.log('执行成功');
} else {
console.log('执行出错');
}
}
Jquery Ajax
说到Jquery,这是一个时代,几乎统治了前端10年有余,彻底解决了UI层与数据层交互的问题,直到三大框架(Angular/React/Vue)的出现,前端进入MVVM浪潮。而Ajax将XHR进行封装,让开发者可以更加便捷方便进行使用。
$.ajax({ //标准写法
type: 'POST',
url: url,
data: data,
dataType: dataType,
success: function () {},
error: function () {}
});
$.get(url,function(){}); //get请求
$.post(url,body,function(){}); //post请求
$.getJSON(url,function(){}); //get请求从服务器加载Json编码
优点:
- 对原生XHR的封装
- 针对MVC的编程
- 完美的兼容性
- 支持jsonp
缺点:
- 不符合MVVM
- 异步模型不够现代,不支持链式,代码可读性差
- 整个Jquery太大,引入成本过高
Fetch
fetch其实是一个新世界,脱离XHR,完全是基于Promise的异步处理机制,使用起来会比ajax更加简单。
使用fetch的代码会相比xhr来说更具有条理性。
fetch(url).then(function(response) {
return response.json();
}).then(function(data) {
console.log(data);
}).catch(function(e) {
console.log("Oops, error");
});
在使用ES6的箭头函数后
fetch(url).then(response => response.json())
.then(data => console.log(data))
.catch(e => console.log("Oops, error", e))
优点:
- 更加底层,提供的API丰富(request, response)
- 语法简单,脱离了XHR,基于ES新的Promise设计
缺点:
- 兼容性比较凄惨,低级别浏览器均不支持,需要实现fetch的polyfill了。思路其实很简单,就是判断浏览器是否支持原生的fetch,不支持的话,就仍然使用XMLHttpRequest的方式实现,同时结合Promise来进行封装。常见的polyfill就有:
es6-promise
,babel-polyfill
,fetch-ie8
等 - 不支持jsonp,可以引入
fetch-jsonp
安装 npm install fetch-jsonp --save-dev
// 使用
fetchJsonp(url, {
timeout: 3000, jsonpCallback: 'callback'
}).then(function(response) {
console.log(response.json());
}).catch(function(e) {
console.log(e)
});
- 没有拦截器,需要额外再封装一层或者
fetch-interceptor
- 默认不带cookie,需要添加配置
fetch(url,{
credentials: 'include'
//include表示cookie既可同域,也可跨域,‘same-origin’表示只可同域
});
- 没有abort,不支持timeout超时处理
可以用Promise.race()实现,Promise.race(iterable) 方法返回一个Promise对象,只要 iterable 中任意一个Promise 被 resolve 或者 reject 后,外部的Promise 就会以相同的值被 resolve 或者 reject。 - 无法获取progress状态
Fetch中的Response.body 中实现了getReader()方法用于读取原始字节流, 该字节流可以循环读取.
Axios
axios也是比较新的网络请求的类库,并且被尤雨溪尤大推荐,已成为VUE的网络请求标配,也是十分的火爆。它本身也是对原生XHR的封装。
- 支持node,创建http请求
- 支持Promise API
- 客户端防止CSRF:每个请求带一个cookie拿到的key
- 拦截请求和响应
- 可取消请求
兼容性上虽然axios本质上也是对原生XHR的封装,但是它也依赖原生ES6 Promise的实现,和fetch一样需要polyfill的兼容。
安装:
//npm
npm install axios
//cdn
基本使用如下:
axios({
method: 'GET',
url: url,
})
.then(res => {console.log(res)})
.catch(err => {console.log(err)})
// get请求
axios.get(url)
.then(function (response) {
console.log(response);
})
.catch(function (error) {
console.log(error);
});
// post请求
axios.post(‘/user’, {
name: 'Jerry',
lastName: 'Liang'
})
.then(function (response) {
console.log(response);
})
.catch(function (error) {
console.log(error);
});
如何选择(个人理解,仅供参考)
- 首先可以肯定的是,如果你的代码依旧是基于Jquery,那毫无疑问,ajax就是你最好的选择。
- 如果你使用的是任意MVVM框架,建议无脑使用axios,fetch在实际项目使用中,需要各种的封装和异常处理,并非开箱即用,而axios可以做到直接替换$.ajax。
- 如果就是要使用fetch,那相信你也一定能封装成自己的一套最佳实践。
参考:https://qianduan.group/posts/5bebe26f9fd64d5a7458a932