JavaScript异步回调的Promise模式封装实例


Posted in Javascript onJune 07, 2014

网页的交互越来越复杂,JavaScript 的异步操作也随之越来越多。如常见的 ajax 请求,需要在请求完成时响应操作,请求通常是异步的,请求的过程中用户还能进行其他的操作,不会对页面进行阻塞,这种异步的交互效果对用户来说是挺有友好的。但是对于开发者来说,要大量处理这种操作,就很不友好了。异步请求完成的操作必须预先定义在回调函数中,等到请求完成就必须调用这个函数。这种非线性的异步编程方式会让开发者很不适应,同时也带来了诸多的不便,增加了代码的耦合度和复杂性,代码的组织上也会很不优雅,大大降低了代码的可维护性。情况再复杂点,如果一个操作要等到多个异步 ajax 请求的完成才能进行,就会出现回调函数嵌套的情况,如果需要嵌套好几层,那你就只能自求多福了。
先看看下面这个常见的异步函数。

var showMsg = function(){
    setTimeout(function(){
        alert( 'hello' );
    }, 5000 );
};

如果要给该函数添加回调,通常会这么干。

var showMsg = function( callback ){
    setTimeout(function(){
        alert( 'hello' );
        // 此处添加回调
        callback();
    }, 5000 );
};

如果是使用 easy.js 的 Promise,添加回调的方法就会优雅多了,前提是需要将原函数封装成一个 promise 实例。

var showMsg = function(){
    // 构造promise实例
    var promise = new E.Promise();    setTimeout(function(){
        alert( 'hello' );
        // 改变promise的状态
        promise.resolve( 'done' );
    }, 5000 );
    // 返回promise实例
    return promise;
};

将一个普通的函数封装成一个 promise 实例,有3个关键步骤,第一步是在函数内部构造一个 promise 实例,第二步是部署函数执行完去改变 promise 的状态为已完成,第三步就是返回这个 promise 实例。每个 promise 实例都有3种状态,分别为 pending(未完成)、resolved(已完成,成功)、rejected(已拒绝,失败)。下面再来看看如何添加回调。

showMsg().then(function( str ){
    // 回调添加到这里来了
    callback( str );
});

这样就将回调函数和原来的异步函数彻底的分离了,从代码组织上看,优雅了很多。resolve 接受一个参数,该参数就可以轻松实现将数据传送给使用 then 方法添加的回调中。
对于 ajax 请求,easy.js 直接将 ajax 方法封装成了 promise 对象,可以直接添加 then 方法来回调。

E.ajax({
    url : 'test1.php',
    type : 'GET'
})
.then(function(){
    // 添加请求成功的回调
}, function(){
    // 添加请求失败的回调
});

then 方法接受2个函数作为参数,第一个函数是已完成的回调,第二个就是已失败的回调。
如果有上面提到的多个 ajax 请求的情况呢?那么就要用到 when 这个方法了。该方法可以接受多个 promise 实例作为参数。

var requests = E.when(E.ajax({
    url : 'test1.php',
    type : 'GET'
}), E.ajax({
    url : 'test2.php',
    type : 'GET'
}));
requests.then(function( arg1, arg2 ){
    console.log( 'success:' + arg1[0] + arg2[0] );
}, function( arg1, arg2 ){
    console.log( 'failure:' + arg1 + arg2  );
});

when 方法是将多个 promise 实例存到一个数组中,等到该数组的所有 promise 实例都是已完成状态才去执行已完成的回调,一旦有一个实例是已拒绝的状态,则立即执行已拒绝的回调。

Promise 模式是 CommonJS 的规范之一。很多主流的 JavaScript 库都有相应的实现,如 jQuery 和 Dojo 中,都有 Deferred 去实现这些功能。在这里还是要吐槽下 jQuery 的 Deferred,撇开其内部使用,这应该用户使用率最低的一个模块了,这和其较复杂的使用方式有一定的关系。

Javascript 相关文章推荐
关于javascript中的parseInt使用技巧
Sep 03 Javascript
javascript 关于# 和 void的区别分析
Oct 26 Javascript
jQuery validate 中文API 附validate.js中文api手册
Jul 31 Javascript
JavaScript实现的一个倒计时的类
Mar 12 Javascript
jQuery添加和删除输入文本框标签代码
May 20 Javascript
JavaScript遍历求解数独问题的主要思路小结
Jun 12 Javascript
jQuery实现鼠标经过像翻页和描点链接效果
Aug 08 Javascript
jQuery EasyUI右键菜单实现关闭标签/选项卡
Oct 10 Javascript
vue+iview+less+echarts实战项目总结
Feb 22 Javascript
默认浏览器设置及vue自动打开页面的方法
Sep 21 Javascript
JS实现滚动条触底加载更多
Sep 19 Javascript
vue父子组件间引用之$parent、$children
May 20 Javascript
jQuery的缓存机制浅析
Jun 07 #Javascript
Firefox中使用outerHTML的2种解决方法
Jun 07 #Javascript
jQuery 顶部导航跟随滚动条滚动固定浮动在顶部
Jun 06 #Javascript
判断iframe里的页面是否加载完成
Jun 06 #Javascript
javascript获取隐藏元素(display:none)的高度和宽度的方法
Jun 06 #Javascript
js中的hasOwnProperty和isPrototypeOf方法使用实例
Jun 06 #Javascript
jquery数组过滤筛选方法grep()简介
Jun 06 #Javascript
You might like
PHP 闭包详解及实例代码
2016/09/28 PHP
Laravel中批量赋值Mass-Assignment的真正含义详解
2017/09/29 PHP
Mac系统下安装PHP Xdebug
2018/03/30 PHP
类似GMAIL的Ajax信息反馈显示
2010/02/16 Javascript
Javascript学习笔记一 之 数据类型
2010/12/15 Javascript
dojo学习第一天 Tab选项卡 实现
2011/08/28 Javascript
Three.js源码阅读笔记(光照部分)
2012/12/27 Javascript
jQuery实现当按下回车键时绑定点击事件
2014/01/28 Javascript
jQuery前端框架easyui使用Dialog时bug处理
2014/12/05 Javascript
js实现图片从左往右渐变切换效果的方法
2015/02/06 Javascript
JavaScript判断FileUpload控件上传文件类型
2015/09/28 Javascript
jquery判断iPhone、Android设备类型
2016/09/14 Javascript
vue实现城市列表选择功能
2018/07/16 Javascript
Vue 实现列表动态添加和删除的两种方法小结
2018/09/07 Javascript
详解JQuery基础动画操作
2019/04/12 jQuery
微信小程序云函数使用mysql数据库过程详解
2019/08/07 Javascript
nodejs文件夹深层复制功能
2019/09/03 NodeJs
[00:36]DOTA2上海特级锦标赛 LGD战队宣传片
2016/03/04 DOTA
在Python中用has_key()方法查找键是否存在的教程
2015/05/21 Python
浅谈编码,解码,乱码的问题
2016/12/30 Python
使用python实现tcp自动重连
2017/07/02 Python
python不换行之end=与逗号的意思及用途
2017/11/21 Python
简单的python协同过滤程序实例代码
2018/01/31 Python
一些Centos Python 生产环境的部署命令(推荐)
2018/05/07 Python
解决python爬虫中有中文的url问题
2018/05/11 Python
python中正则表达式 re.findall 用法
2018/10/23 Python
用python代码将tiff图片存储到jpg的方法
2018/12/04 Python
基于django channel实现websocket的聊天室的方法示例
2019/04/11 Python
Python如何实现动态数组
2019/11/02 Python
Pycharm创建项目时如何自动添加头部信息
2019/11/14 Python
Python sublime安装及配置过程详解
2020/06/29 Python
乔丹诺(Giordano)酒庄德国官网:找到最好的意大利葡萄酒
2017/12/28 全球购物
介绍信模板
2015/01/31 职场文书
《月球之谜》教学反思
2016/02/20 职场文书
vue自定义右键菜单之全局实现
2022/04/09 Vue.js
Redis 报错 error:NOAUTH Authentication required
2022/05/15 Redis