调用 API 访问数据采用的 Ajax 方式,这是一个异步过程,异步过程最基本的处理方式是事件或回调,其实这两种处理方式实现原理差很少,都须要在调用异步过程的时候传入一个在异步过程结束的时候调用的接口。好比 jQuery Ajax 的 success
就是典型的回调参数。不过使用 jQuery 处理异步推荐使用 Promise 处理方式。前端
Promise 处理方式也是经过注册回调函数来完成的。jQuery 的 Promise 和 ES6 的标准 Promise 有点不同,但在 then
上能够兼容,一般称为 thenable。jQuery 的 Promise 没有提供 .catch()
接口,但它本身定义的 .done()
、.fail()
和 .always()
三个注册回调的方式也颇有特点,用起来很方便,它是在事件的方式来注册的(即,能够注册多个同类型的处理函数,在该触发的时候都会触发)。ios
固然更直观的一点的处理方式是使用 ES2017 带来的 async/await 方式,能够用同步代码的形式来写异步代码,固然也有一些坑在里面。对于前端工程师来讲,最大的坑就是有些浏览器不支持,须要进行转译,因此若是前端代码没有构建过程,通常仍是就用 ES5 的语法兼容性好一些(jQuery 的 Promise 是支持 ES5 的,可是标准 Promise 要 ES6 之后才可使用)。ajax
关于 JavaScript 异步处理相关的内容能够参考json
闲谈异步调用“扁平”化segmentfault
理解 JavaScript 的 async/awaitpromise
在处理 Ajax 的过程当中,虽然有现成的库(好比 jQuery.ajax,axios 等),它毕竟是为了通用目的设计的,在使用的时候仍然难免繁琐。而在项目中,对 Api 进行调用的过程几乎都大同小异。若是设计得当,就连错误处理的方式都会是同样的。所以,在项目内的 Ajax 调用其实能够进行进一步的封装,使之在项目内使用起来更方便。若是接口方式发生变化,修改起来也更容易。服务器
好比,当前接口要求使用 POST 方法调用(暂不考虑 RESTful),参数必须包括 action
,返回的数据以 JSON 方式提供,若是出错,只要不是服务器异常都会返回特定的 JSON 数据,包括一个不等于 0 的 code
和可选的 message
属性。
那么用 jQuery 写这么一个 Ajax 调用,大概是这样
const apiUrl = "http://api.some.com/"; jQuery .ajax(url, { type: "post", dataType: "json", data: { action: "login", username: "uname", password: "passwd" } }) .done(function(data) { if (data.code) { alert(data.message || "登陆失败!"); } else { window.location.assign("home"); } }) .fail(function() { alert("服务器错误"); });
同一项目中,这样的 Ajax 调用,基本上只有 data
部分和 .done
回调中的 else
部分不一样,因此进行一次封装会大大减小代码量,能够这样封装
function appAjax(action, params) { var deffered = $.Deferred(); jQuery .ajax(apiUrl, { type: "post", dataType: "json", data: $.extend({ action: action }, params) }) .done(function(data) { // 当 code 为 0 或省略时,表示没有错误, // 其它值表示错误代码 if (data.code) { if (data.message) { // 若是服务器返回了消息,那么向用户呈现消息 // resolve(null),表示不须要后续进行业务处理 alert(data.message); deffered.resolve(); } else { // 若是服务器没返回消息,那么把 data 丢给外面的业务处理 deferred.reject(data); } } else { // 正常返回数据的状况 deffered.resolve(data); } }) .fail(function() { // Ajax 调用失败,向用户呈现消息,同时不须要进行后续的业务处理 alert("服务器错误"); deffered.resolve(); }); return deferred.promise(); }
而业务层的调用就很简单了
appAjax("login", { username: "uname", password: "passwd" }).done(function(data) { if (data) { window.location.assign("home"); } }).fail(function() { alert("登陆失败"); });
上面的封装对调用接口和返回数据进行了统一处理,把大部分项目接口约定的内容都处理掉了,剩下在每次调用时须要处理的就是纯粹的业务。
如今项目组决定不用 jQuery 的 Ajax,而是采用 axios 来调用 API(axios 不见得就比 jQuery 好,这里只是举例),那么只须要修改一下 appAjax()
的实现便可。全部业务调用都不须要修改。
假设如今的目标环境仍然是 ES5,那么须要第三方 Promise 提供,这里拟用 Bluebird,兼容原生 Promise 接口(在 HTML 中引入,未直接出如今 JS 代码中)。
function appAjax(action, params) { var deffered = $.Deferred(); axios .post(apiUrl, { data: $.extend({ action: action }, params) }) .then(function(data) { ... }, function() { ... }); return deferred.promise(); }
此次的封装采用了 axios 来实现 Web Api 调用。可是为了保持原来的接口(jQuery Promise 对象有提供 .done()
、.fail()
和 .always()
事件处理),appAjax
仍然不得不返回 jQuery Promise。这样,即便全部地方都再也不须要使用 jQuery,这里仍然得用。
项目中应该用仍是不用 jQuery?请阅读为何要用原生 JavaScript 代替 jQuery?
就只在这里使用 jQuery 总让人感受如芒在背,想把它去掉。有两个办法
修改全部业务中的调用,去掉 .done()
、.fail()
和 .always()
,改为 .then()
。这一步工做量较大,但基本无痛,由于 jQuery Promise 自己支持 .then()
。可是有一点须要特别注意,这一点稍后说明
本身写个适配器,兼容 jQuery Promise 的接口,工做量也不小,但关键是要充分测试,避免差错。
上面提到第 1 种方法中有一点须要特别注意,那就是 .then()
和 .done()
系列函数在处理方式上有所不一样。.then()
是按 Promise 的特性设计的,它返回的是另外一个 Promise 对象;而 .done()
系列函数是按事件机制实现的,返回的是原来的 Promise 对象。因此像下面这样的代码在修改时就要注意了
appAjax(url, params) .done(function(data) { console.log("第 1 到处理", data) }) .done(function(data) { console.log("第 2 到处理", data) }); // 第 1 到处理 {} // 第 2 到处理 {}
简单的把 .done()
改为 .then()
以后(注意不须要使用 Bluebird,由于 jQuery Promise 支持 .then()
)
appAjax(url, params) .then(function(data) { console.log("第 1 到处理", data); }) .then(function(data) { console.log("第 2 到处理", data); }); // 第 1 到处理 {} // 第 2 到处理 undefined
缘由上面已经讲了,这里正确的处理方式是合并多个 done 的代码,或者在 .then()
处理函数中返回 data
:
appAjax(url, params) .then(function(data) { console.log("第 1 到处理", data); return data; }) .then(function(data) { console.log("第 2 到处理", data); });
咱们的 appAjax()
接口部分也能够设计成 Promise 实现,这是一个更通用的接口。既使用不用 ES2015+ 特性,也可使用像 jQuery Promise 或 Bluebird 这样的三方库提供的 Promise。
function appAjax(action, params) { // axios 依赖于 Promise,ES5 中可使用 Bluebird 提供的 Promise return axios .post(apiUrl, { data: $.extend({ action: action }, params) }) .then(function(data) { // 这里调整了判断顺序,会让代码看起来更简洁 if (!data.code) { return data; } if (!data.message) { throw data; } alert(data.message); }, function() { alert("服务器错误"); }); }
不过如今前端有构建工具,可使用 ES2015+ 配置 Babel,也可使用 TypeScript …… 总之,选择不少,写起来也很方便。那么在设计的时候就不用局限于 ES5 所支持的内容了。因此能够考虑用 Promise + async/await 来实现
async function appAjax(action, params) { // axios 依赖于 Promise,ES5 中可使用 Bluebird 提供的 Promise const data = await axios .post(apiUrl, { data: $.extend({ action: action }, params) }) // 这里模拟一个包含错误消息的结果,以便后面统一处理错误 // 这样就不须要用 try ... catch 了 .catch(() => ({ code: -1, message: "服务器错误" })); if (!data.code) { return data; } if (!data.message) { throw data; } alert(data.message); }
上面代码中使用
.catch()
来避免try ... catch ...
的技巧在从不用 try-catch 实现的 async/await 语法说错误处理中提到过。
固然业务层调用也可使用 async/await(记得写在 async 函数中):
const data = await appAjax("login", { username: "uname", password: "passwd" }).catch(() => { alert("登陆失败"); }); if (data) { window.location.assign("home"); }
对于屡次 .done()
的改造:
const data = await appAjax(url, params); console.log("第 1 到处理", data); console.log("第 2 到处理", data);
本文以封装 Ajax 调用为例,看似在讲述异步调用。但实际想告诉你们的东西是:如何将一个经常使用的功能封装起来,实现代码重用和更简洁的调用;以及在封装的过程当中须要考虑的问题——向前和向后的兼容性,在作工具函数封装的时候,应该尽可能避免和某个特定的工具特性绑定,向公共标准靠拢——不知你们是否有所体会。