JavaScript中Promises/A+规范的实现
Promises是一种异步编程模型,通过一组API来规范化异步操作,这样也能够让异步操作的流程控制更加容易。
下面的代码是假设执行一个异步队列,每一项都会使用上一项返回的数据:
function nest(url, params, fn) {
$.getJSON(url, params, function(data) {
console.log(data);
fn.call(this);
});
}
nest('promise.php', {a:1}, function(data1) {
nest('promise.php', {b:2}, function(data2) {
nest('promise.php', {c:3}, function(data3) {
console.log('同步完成');
});
});
});
这是一个回调金字塔,当异步的任务很多的时候,需要维护大量的callback。这嵌套的自己眼睛都看不清了吧。
Promise/A+规范就是为了解决上面的问题,可以用类似下面的代码来改进,“nest”也会做些修改:
promise.then(nest('promise.php', {a:1}))
.then(nest('promise.php', {b:2}))
.then(nest('promise.php', {c:3}));
下图是改进的图片示例,我在画图的时候也感觉到,左边的比较难画,右边的很好画。

接下来会围绕改进金字塔,实现规范展开。
一、Promises/A+规范说明
1)一个promise可能有三种状态:等待(pending)、已完成(fulfilled)、已拒绝(rejected)
2)一个promise的状态只可能从“等待”转到“完成”态或者“拒绝”态,不能逆向转换,同时“完成”态和“拒绝”态不能相互转换
3)promise必须实现then方法,而且then必须返回一个promise
4)同一个promise的then可以调用多次,并且回调的执行顺序跟它们被定义时的顺序一致
5)then方法接受两个参数,第一个参数是成功时的回调,在promise由“等待”态转换到“完成”态时调用
6)另一个是失败时的回调,在promise由“等待”态转换到“拒绝”态时调用。
7)then可以接受另一个promise传入,也接受一个“类then”的对象或方法,即thenable对象。
接下来先实现一个简易的,只有完成状态,没有拒绝和等待状态。
二、简单实现

Promise对象的实现:
function Promise(fn) {
this._status = 'pending';
this._resolves = []; //队列
this._fn = fn;
return this;
}
Promise.prototype = {
then: function(resolve) {
var next = this._next || (this._next = new Promise()); //下一个promise对象
this._resolves.push(resolve); //设置队列
return next;
},
resolved: function(value) { //改变状态
this._status = 'fulfilled';
this._result = (this._fn && this._fn(value)) || value;
while (fn = this._resolves.shift()) { //循环调用队列
this._fire(this._next, fn);
}
},
_fire: function(nextPromise, nextFn) {
var nextResult = nextFn(this._result);
if (nextResult instanceof Promise) { //判断回调是否是Promise对象
//只有当nextResult的状态为fulfilled,下一个promise才可以执行
nextResult.then(function(value) {
nextPromise.resolved(value);
});
} else {
nextPromise.resolved(nextResult);
}
}
};
演示用的函数:
function nest2(url, params) {
return function(pre) {
var promise = new Promise();
$.getJSON(url, params, function(data) {
promise.resolved(data);
});
return promise;
};
}
function begin(value) {
return value + '!';
}
初始化代码:
var promise = new Promise(begin);
promise.then(nest2('promise.php', {a: 1}))
.then(nest2('promise.php', {b: 2}));
promise.resolved('开始');
也可以另外一种方式调用,这样的话内部的_resloves队列中会有多个值
var promise = new Promise(begin);
promise.then(nest2('promise.php', {a: 1}))
promise.then(nest2('promise.php', {b: 2}));
promise.resolved('开始');
demo下载:
http://download.csdn.net/detail/loneleaf1/9391315
参考资料:
http://www.alloyteam.com/2014/05/javascript-promise-mode/ JavaScript Promise启示录
http://www.cnblogs.com/fsjohnhuang/p/4135149.html JS魔法堂:剖析源码理解Promises/A规范
http://www.cnblogs.com/aaronjs/archive/2012/11/17/2774440.html 使用Promises/A
http://rapheal.sinaapp.com/2013/01/26/jquery-src-deferred/ $.Deferred
http://www.ituring.com.cn/article/66566 Promises/A+规范
JavaScript中Promises/A+规范的实现的更多相关文章
- 细谈JavaScript中的书写规范
当你有一些感触想写下一些东西的时候,总会发现其实你想写的所有文章都有大牛已经给你写好了,而且写的比你好. https://github.com/ecomfe/spec/blob/master/java ...
- JavaScript 中语法规范及调试
JavaScript 中语法规范及调试 版权声明:未经博主授权,内容严禁分享转载 JavaScript 开发环境 JavaScript 脚本可以使用任意一款纯文本编辑器进行编程开发. 常见的前端开发编 ...
- CommonJS Promises/A规范
本文来自四火哥的翻译 CommonJS是一组javascript编程规范,而promise是其中之一. 简而言之,promises是一种令代码的异步行为变得更加优雅的软件抽象.在基本的定义中,代码可能 ...
- 通过一道笔试题浅谈javascript中的promise对象
因为前几天做了一个promise对象捕获错误的面试题目,所以这几天又重温了一下promise对象.现在借这道题来分享下一些很基础的知识点. 下面是一个面试题目,三个promise对象捕获错误的例子,返 ...
- JS魔法堂:剖析源码理解Promises/A规范
一.前言 Promises/A是由CommonJS组织制定的异步模式编程规范,有不少库已根据该规范及后来经改进的Promises/A+规范提供了实现 如Q, Bluebird, when, rsvp. ...
- 前端翻译:Promises/A+规范
原文地址:https://promisesaplus.com/ 本篇为原文翻译+个人理解,若有谬误请各位指正,谢谢. 尊重原创,转载请注明来自:http://www.cnblogs.com/fsjoh ...
- JavaScript中模块“写法”
在JavaScript模块到底是什么 event = function() { // do more return { bind: function() {}, unbind: function() ...
- 对 Promises/A+ 规范的研究 ------引用
作为 Modern JavaScript 基础设施的一部分,Promises 对前端开发者而言异常重要.它是 async/await 语法的基础,是 JavaScript 中处理异步的标准形式.并且, ...
- [技术翻译]在现代JavaScript中编写异步任务
本周再来翻译一些技术文章,本次预计翻译三篇文章如下: 04.[译]使用Nuxt生成静态网站(Generate Static Websites with Nuxt) 05.[译]Web网页内容是如何影响 ...
随机推荐
- Web前端面试之HTML
1. 对WEB标准以及W3C的理解与认识 web标准规范要求,书写标签闭合.小写.不乱嵌套,可提高搜索机器人对网页内容的搜索几率.--- SEO 使用外链css和js脚本,结构与行为.结构与表现分离, ...
- Android :fragment介绍
一.关于Fragmemt 1.Fragment(片段),主要是为了支持更多的动态和灵活的用户界面设计,如平板电脑.Fragment允许组合和交换用户界面组件,而不需要更改视图层次结构.通过把Activ ...
- MLlib决策树与集成树
决策树是一种常见的分类与回归机器学习算法,由于其模型表达性好,便于理解,并能取得较好的效果,而受到广泛的应用.下图是一个简单的决策树,决策树每个非叶子节点包含一个条件,对于具有连续值的特征,该条件为一 ...
- Android进程间通讯之messenger
这两天在看binder,无意间在文档看到messenger这么个东西,感觉这个东西还挺有意思的,给大家分享一下. 平时一说进程间通讯,大家都会想到AIDL,其实messenger和AIDL作用一样,都 ...
- spring+mybatis+oracle/mysql整合开发需要的jar包详解
导入spring,mybatis,c3p0,oracle和mybatis提供的与spring整合的插件包 mysql的jar: mysql-connector-java-5.1.7 ...
- 奇怪的UnexpectedRollbackException异常
今天在使用一个原来常用的功能的时候,突然发现在某些场景下会报异常,内容如下: 通过断点调试发现一路都很顺畅,就是在从controller层返回前段的时候会报该异常,没办法,只能通过排除法定位问题,后来 ...
- 一鼓作气 博客--第七篇 note7
面向对象相关知识简介 类(Class): 用来描述具有相同的属性和方法的对象的集合.它定义了该集合中每个对象所共有的属性和方法.对象是类的实例. 类变量:类变量在整个实例化的对象中是公用的.类变量定义 ...
- 夜深了,写了个JQuery的省市区三级级联效果
刚刚练手的JQuery,希望大神们指正 主要实现以下功能: 1.三级菜单级联加载数据 2.可以在不操作脚本的情况下,给元素加属性实现级联功能 3.自定义动态显示数据 咨询问题: 对于一般比较固定不变的 ...
- SQL SERVER全面优化-------写出好语句是习惯
前几篇文章已经从整体提供了诊断数据库的各个方面问题的基本思路...也许对你很有用,也许你觉得离自己太远.那么今天我们从语句的一些优化写法及一些简单优化方法做一个介绍.这对于很多开发人员来说还是很有用的 ...
- 剑指Offer面试题:18.二叉树的镜像
一.题目:二叉树的镜像 题目:请完成一个函数,输入一个二叉树,该函数输出它的镜像.例如下图所示,左图是原二叉树,而右图则是该二叉树的镜像. 该二叉树节点的定义如下,采用C#语言描述: public c ...