JavaScript的Callback机制深入人心。而ECMAScript的世界同样充斥的各种异步操作(异步IO、setTimeout等)。异步和Callback的搭载很容易就衍生"回调金字塔"。——由此产生Deferred/Promise。

Deferred起源于Python,后来被CommonJS挖掘并发扬光大,得到了大名鼎鼎的Promise,并且已经纳入ECMAScript 6(JavaScript下一版本)。

Promise/Deferred是当今最著名的异步模型,不仅强壮了JavaScript Event Loop(事件轮询)机制下异步代码的模型,同时增强了异步代码的可靠性。—— 匠者为之,以惠匠者。

> 本文内容如下:
>
> - Promise应对的问题
> - Promise的解决
> - ECMAScript 6 Promise
> - 参考和引用

Promise应对的问题

JavaScript充斥着Callback,例如下面的代码:

  1. (function (num) {//从外面接收一个参数
  2. var writeName = function (callback) {
  3. if (num === 1)
  4. callback();
  5. }
  6. writeName(function () {//callback
  7. console.log("i'm linkFly");
  8. });
  9. })(1);

把一个函数通过参数传递,那么这个函数叫做Callback(回调函数)。

JavaScript也充斥着异步操作——例如ajax。下面的代码就是一段异步操作:

  1. var name;
  2. setTimeout(function () {
  3. name = 'linkFly';
  4. }, 1000);//1s后执行
  5. console.log(name);//输出undefined

这段代码的运行逻辑是这样的:

我们的总是遇见这样的情况:一段代码异步执行,后续的代码却需要等待异步代码的,如果在异步代码之前执行,就会如上面的console.log(name)一样,输出undefined,这并不是我们想要的效果。

类似的情况总是发生在我们经常要使用的ajax上:

  1. $.ajax({
  2. url: 'http://www.cnblogs.com/silin6/map',
  3. success: function (key) {
  4. //我们必须要等待这个ajax加载完成才能发起第二个ajax
  5. $.ajax({
  6. url: 'http://www.cnblogs.com/silin6/source/' + key,
  7. success: function (data) {
  8. console.log("i'm linkFly");//后输出
  9. }
  10. });
  11. }
  12. });
  13. console.log('ok');//ok会在ajax之前执行

异步操作有点类似这一段代码被挂起,先执行后续的代码,直到异步得到响应(例如setTimeout要求的1s之后执行,ajax的服务器响应),这一段异步的代码才会执行。关于这一段异步代码的执行流程,请参阅JavaScript大名鼎鼎的:Event Loop(事件轮询)

Promise的解决

Promise优雅的修正了异步代码,我们使用Promise重写我们setTimeout的示例:

  1. var name,
  2. p = new Promise(function (resolve) {
  3. setTimeout(function () {//异步回调
  4. resolve();
  5. }, 1000);//1s后执行
  6. });
  7. p.then(function () {
  8. name = 'linkFly';
  9. console.log(name);//linkFly
  10. }).then(function () {
  11. name = 'cnBlog';
  12. console.log(name);
  13. });
  14. //这段代码1s后会输出linkFly,cbBlog

我们先不要太过在意Promise对象的API,后续会讲解,我们只需要知道这段代码完成了和之前同样的工作。我们的console.log(name)正确的输出了linkFly,并且我们还神奇的输出了cnBlog。

或许你觉得这段代码实在繁琐,还不如setTimeout来的痛快,那么我们再来改写上面的ajax:

  1. var ajax = function (url) {
  2. //我们改写ajax,让它以Promise的方式工作
  3. return new Promise(function (resolve) {
  4. $.ajax({
  5. url: url,
  6. success: function (data) {
  7. resolve(data);
  8. }
  9. });
  10. });
  11. };
  12. ajax('http://www.cnblogs.com/silin6/map')
  13. .then(function (key) {
  14. //我们得到key,发起第二条请求
  15. return ajax('http://www.cnblogs.com/silin6/source/' + key);
  16. })
  17. .then(function (data) {
  18. console.log(data);//这时候我们会接收到第二次ajax返回的数据
  19. });

或许它晦涩难懂,那么我们尝试用setTimeout来模拟这次的ajax,这个例子演示了Promise数据的传递,一如ajax:

  1. var name,
  2. ajax = function (data) {
  3. return new Promise(function (resolve) {
  4. setTimeout(function () {//我们使用setTimeout模拟ajax
  5. resolve(data);
  6. }, 1000);//1s后执行
  7. });
  8. };
  9. ajax('linkFly').then(function (name) {
  10. return ajax("i'm " + name);//模拟第二次ajax
  11. }).then(function (value) {
  12. //2s后,输出i'm linkFly
  13. console.log(value);
  14. });

上面的代码,从代码语义上达到了下面的流程:

我们仅观察代码就知道现在的它变得非常优雅,两次异步的代码被完美的抹平。但我们应该时刻谨记,Promise改变的是你异步的代码和编程思想,而并没有改变异步代码的执行——它是一种由卓越的编程思想所衍生的对象。

下面一张图演示了普通异步回调和Promise异步的区别,Promise实现的异步从代码运行上来说并无太大区别,但从编程思想上来说差异巨大。

ECMAScript 6 Promise##

Promise对象代表了未来某个将要发生的事件(通常是一个异步操作),抹平了异步代码的金字塔,它从模型上解决了异步代码产生的"回调金字塔"。

Promise是ECMAScript 6规范内定义的,所以请使用现代浏览器测试,它的兼容性可以在这里查看

Promise.constructor

Promise是一个对象,它的构造函数接收一个回调函数,这个回调函数参数有两个函数:分别在成功状态下执行和失败状态下执行,Promise有三个状态,分别为:等待态(Pending)、执行态(Fulfilled)和拒绝态(Rejected)。

  1. var p = new Promise(function (resolve,reject) {
  2. console.log(arguments);
  3. //resolve表示成功状态下执行
  4. //reject表示失败状态下执行
  5. });

传递的这个回调函数,等同被Promise重新封装,并传递了两个参数回调,这两个参数用于驱动Promise数据的传递。resolve和reject本身承载着触发器的使命:

  • 默认的Promise对象是等待态(Pending)
  • 调用resolve()表示这个Promise进入执行态(Fulfilled)
  • 调用reject()表示这个promise()进入拒绝态(Rejected)
  • Promise对象可以从等待状态下进入到执行态和拒绝态,并且无法回退。
  • 而执行态和拒绝态不允许互相转换(例如执行态转换到拒绝态)。

Promise.prototype.then

生成的promise实例(如上面的变量p)拥有方法then(),then()方法是Promise对象的核心,它返回一个新的Promise对象,因此可以像jQuery一样链式操作,非常优雅。

Promise是双链的,所以then()方法接受两个参数,分别表示:

  • _执行态(Fulfilled)_下执行的回调函数
  • _拒绝态(Rejected)_下执行的回调函数。
  1. p.then(function () {
  2. //我们返回一个promise
  3. return new Promise(function (resolve) {
  4. setTimeout(function () {
  5. resolve('resolve');
  6. }, 1000);//异步1s
  7. });
  8. }, function () {
  9. console.log('rejected');
  10. }) //链式回调
  11. .then(function (state) {
  12. console.log(state);//如果为执行态,输出resolve
  13. }, function (data) {
  14. console.log(data);//如果为拒绝态,输出undefined
  15. });;

then()方法的返回值由它相应状态下执行的函数决定:这个函数返回undefined,则then()方法构建一个默认的Promise对象,并且这个对象拥有then()方法所属的Promise对象的状态。

  1. var p = new Promise(function (resolve) {
  2. resolve();//直接标志执行态
  3. }), temp;
  4. temp = p.then(function () {
  5. //传入执行态函数,不返回值
  6. });
  7. temp.then(function () {
  8. console.log('fulfilled');//拥有p的状态
  9. });
  10. console.log(temp === p);//默认构建的promise,但已经和p不是同一个对象,输出false

如果对应状态所执行的函数返回一个全新的Promise对象,则会覆盖掉当前Promise,代码如下:

  1. var p = new Promise(function (resolve) {
  2. resolve();//直接标志执行态
  3. }), temp;
  4. temp = p.then(function () {
  5. //返回新的promise对象,和p的状态无关
  6. return new Promise(function (resolve, reject) {
  7. reject();//标志拒绝态
  8. });
  9. });
  10. temp.then(function () {
  11. console.log('fulfilled');
  12. }, function () {
  13. console.log('rejected');//输出
  14. });

即then()方法传递的进入的回调函数,如果返回promise对象,则then()方法返回这个promise对象,否则将默认构建一个新的promise对象,并继承调用then()方法的promise的状态。

我们应该清楚Promise的使命,抹平了异步代码的回调金字塔,我们会有很多依赖上一层异步的代码:

  1. var url = 'http://www.cnblogs.com/silin6/';
  2. ajax(url, function (data) {
  3. ajax(url + data, function (data2) {
  4. ajax(url + data2, function (data3) {
  5. ajax(url + data3, function () {
  6. //回调金字塔
  7. });
  8. });
  9. });
  10. });

使用Promise则抹平了代码:

  1. promise.then(function (data) {
  2. return ajax(url + data);
  3. }).then(function (data2) {
  4. return ajax(url + data2);
  5. }).then(function (data3) {
  6. return ajax(url + data3);
  7. }).then(function (data) {
  8. //扁平化代码
  9. });

Promise还有更多更强大的API。但本文的目的旨在让大家感受到Promise的魅力,而并非讲解Promise对象自身的API,关于Promise其他辅助实现API请查阅本文最下方的引用章节,Promise其他API如下:

  • Promise.prototype.catch():用于指定发生错误时的回调函数(捕获异常),并具有冒泡性质。
  • Promise.all()Promise.race():Promise.all方法用于将多个Promise实例,包装成一个新的Promise实例。
  • Promise.resolve()Promise.reject():将现有对象转为Promise对象。

希望大家一点点的接受Promise,所以没有讲太多,我们对于Promise的理解不应该仅仅是一个异步模型,我们更关注应该是Promise/Deferred的编程思想,所以后续几篇会逐渐深入讲解Promise的前生今世。

参考和引用##

作者:linkFly
声明:嘿!你都拷走上面那么一大段了,我觉得你应该也不介意顺便拷走这一小段,希望你能够在每一次的引用中都保留这一段声明,尊重作者的辛勤劳动成果,本文与博客园共享。

JavaScript异步编程(1)- ECMAScript 6的Promise对象的更多相关文章

  1. JavaScript 异步编程(二):Promise

    PromiseState Promise 有一个 [[PromiseState]] 属性,表示当前的状态,状态有 pending 和 fulfill 以及 reject. 从第一个 Promise 开 ...

  2. javascript异步编程,promise概念

    javascript 异步编程 概述 采用单线程模式工作的原因: 避免多线dom操作同步问题,javascript的执行环境中负责执行代码的线程只有一个 内容概要 同步模式和异步模式 事件循环和消息队 ...

  3. JavaScript异步编程原理

    众所周知,JavaScript 的执行环境是单线程的,所谓的单线程就是一次只能完成一个任务,其任务的调度方式就是排队,这就和火车站洗手间门口的等待一样,前面的那个人没有搞定,你就只能站在后面排队等着. ...

  4. JavaScript异步编程(2)- 先驱者:jsDeferred

    JavaScript当前有众多实现异步编程的方式,最为耀眼的就是ECMAScript 6规范中的Promise对象,它来自于CommonJS小组的努力:Promise/A+规范. 研究javascri ...

  5. JavaScript异步编程的主要解决方案—对不起,我和你不在同一个频率上

    众所周知(这也忒夸张了吧?),Javascript通过事件驱动机制,在单线程模型下,以异步的形式来实现非阻塞的IO操作.这种模式使得JavaScript在处理事务时非常高效,但这带来了很多问题,比如异 ...

  6. javascript异步编程的前世今生,从onclick到await/async

    javascript与异步编程 为了避免资源管理等复杂性的问题, javascript被设计为单线程的语言,即使有了html5 worker,也不能直接访问dom. javascript 设计之初是为 ...

  7. Promises与Javascript异步编程

    Promises与Javascript异步编程 转载:http://www.zawaliang.com/2013/08/399.html 在如今都追求用户体验的时代,Ajax应用真的是无所不在.加上这 ...

  8. 5分种让你了解javascript异步编程的前世今生,从onclick到await/async

      javascript与异步编程 为了避免资源管理等复杂性的问题,javascript被设计为单线程的语言,即使有了html5 worker,也不能直接访问dom. javascript 设计之初是 ...

  9. 转: Promises与Javascript异步编程

    在如今都追求用户体验的时代,Ajax应用真的是无所不在.加上这些年浏览器技术.HTML5以及CSS3等的发展,越来越多的富Web应用出现:在给与我们良好体验的同时,Web开发人员在背后需要处理越来越多 ...

随机推荐

  1. OpenCascade B-Spline Basis Function

    OpenCascade B-Spline Basis Function eryar@163.com Abstract. B-splines are quite a bit more flexible ...

  2. 深入理解javascript作用域系列第五篇——一张图理解执行环境和作用域

    × 目录 [1]图示 [2]概念 [3]说明[4]总结 前面的话 对于执行环境(execution context)和作用域(scope)并不容易区分,甚至很多人认为它们就是一回事,只是高程和犀牛书关 ...

  3. php教程|php基础知识

    第1章  初识PHP 当前网络技术发展日新月异,各种基于服务端创建动态网站的脚本语言更是层出不穷.其中PHP以其简单.易用.可移植性强等特点,在众多的动态网站语言技术中独树一帜.那么到底什么是PHP, ...

  4. C语言 第六章 多重循环

    一.概要 在c语言中,if,switch,for,while,do-while可以相互间多次嵌套. if(){ for() { for() { } } } while() { for(){ } for ...

  5. 一种C#生成符合Java规则的二进制文件方法

    文章版权由作者李晓晖和博客园共有,若转载请于明显处标明出处:http://www.cnblogs.com/naaoveGIS/. 1.一个项目中的真实问题 实际项目中,本想通过C#制作小工具生成SHP ...

  6. C#枚举类型的常用操作总结

    枚举类型是定义了一组"符号名称/值"配对.枚举类型是强类型的.每个枚举类型都是从system.Enum派生,又从system.ValueType派生,而system.ValueTy ...

  7. jquery 图片轮播demo实现

    转载注明出处!!! 转载注明出处!!! 转载注明出处!!! 图片轮播demo,弄清楚过程其实是一个很简单的东西,看网上都没有什么实质性的代码,就自己把过程捋了一遍实现了. 这次因为随手写的,所以没有做 ...

  8. DX9入门笔记1-D3D初始化

    对3D编程期待已久,却一直叶公好龙浅尝辄止.近期在公司实习却无具体的工作安排,琢磨着学习个新的手艺,就又想起了3D Programming.这次从大名鼎鼎的龙书(Introduction to 3D ...

  9. C++异常处理: try,catch,throw,finally的用法

    写在前面 所谓异常处理,即让一个程序运行时遇到自己无法处理的错误时抛出一个异常,希望调用者可以发现处理问题. 异常处理的基本思想是简化程序的错误代码,为程序键壮性提供一个标准检测机制. 也许我们已经使 ...

  10. 各种类型转换为字符串类型(ToString())

    更详细请参考:http://blog.csdn.net/wanzhuan2010/article/details/8478904 // C 货币 2.5.ToString("C") ...