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网页内容是如何影响 ...
随机推荐
- javascript 原型及原型链的初步理解
最近折腾了好久,终于是把js里面的原型和原型链做了个初步的理解: 在这里,我打个比喻: 我(child),我妈constructor(构造函数)生了我:别人问我老妈跟谁生的我,于是此时我妈会指向我爸爸 ...
- eclipse插件开发入门
2016-09-09 17:11:50 1. 概述 1.1 SWT/JFace 是Eclipse 的基础,Eclipse 的 Workbench 就是建立在 SWT/JFace 之上的.另外,JFac ...
- JQuery 获取父级元素、同级元素、子元素等
例: <div> <div id="div_1">这是内容1</div> <div id="div_2">这是内 ...
- SQL Server2005清除数据库日志
SQL2005清空删除日志: 复制代码 代码如下: Backup Log DNName with no_log '这里的DNName是你要收缩的数据库名,自己注意修改下面的数据库名,我就不再注释了 ...
- C语言中的插入排序(2016-12-30)
直接插入排序: 算法思想:假设待排序的记录存放在数组R[1--n]中,初始时,i=1,R[1]自成一个有序区,无序区为R[2--n].然后从i=2起直到i=n,依次将R[i]插入当前的有序区R[1.. ...
- 减小ipa体积之删除frameWork中无用mach-O文件
最近项目末期, 我们团队为了ipa的大小使用不少的体积减小的方法, 除了一些常规的方法之外, 我分享一下自己研究出来的新思路. 首先我们来简单的介绍一下mach-O. 什么是mach-O? Mach- ...
- Lesson 7 Too late
Text The plane was late and detectives were waiting at the airport all morning. They were expecting ...
- 理解C# 4 dynamic(3) – DynamicObject的使用
上篇文章"理解C# 4 dynamic(2) – ExpandoObject的使用" 了解了ExpandoObject的基本使用. 但ExpandoObject的问题就是它是一个万 ...
- iOS开发系列--C语言之指针
概览 指针是C语言的精髓,但是很多初学者往往对于指针的概念并不深刻,以至于学完之后随着时间的推移越来越模糊,感觉指针难以掌握,本文通过简单的例子试图将指针解释清楚,今天的重点有几个方面: 什么是指针 ...
- Win8 Metro动态加载内容框架
制作背景 为了参加ImagineCup 2013 世界公民类比赛,我们设计制作了一个可动态扩展的幼教类App.这个App需要能动态加载内容,内容包括带动画可交互的电子书,动画,视频,游戏. 技术支持 ...