深入解析 Promise 的行为

属性和方法

Promise 构造函数接受一个函数作为参数,该函数的两个参数分别是 resolve 和 reject。
Promise 有两个内置属性,[[PromiseStatus]] 和 [[PromiseValue]]。

  • [[PromiseStatus]]:Promise 对象的状态,这个属性可以为三个值:resolved、pending 和 rejected,状态一旦改变为 resolved 或者 rejected 就不能够再变回 pending,resolved 和 rejected 也不能够互相转化。
  • [[PromiseStatus]]:是 Promise 的值,这个值为传入 resolve() 方法和 reject() 方法中的参数。

Promise 对象有两个方法,then 方法和 catch 方法。

  • then():可以接受两个回调函数作为参数。第一个回调函数在 Promise 对象的状态变为 resolved 时调用,第二个回调函数在 Promise 对象的状态变为 rejected 时调用。这两个函数都接受 [[PromiseStatus]] 作为参数。
  • catch():.then(null, rejection) 的别名,用于指定发生错误时的回调函数。

Promise 构造函数的方法:

  • Promise.resolve():接受一个参数,创建一个新的 Promise 对象,[[PrmiseStatus]] 为 resolved,[[PromiseValue]] 为传入的参数。
  • Promise.reject():类似 Promise.resolve()。
  • Promise.all():接受一个数组作为参数,将其中的多个 Promise 对象包装成一个新的 Promise 对象。如果数组中某个元素不是 Promise 对象,那么将这个参数传入 Promise.resolve() 方法转换为 Promise 对象。只要有一个 Promise 对象的状态变为 rejected,新 Promise 状态就会立刻变为 rejected,只有当所有的 Promise 对象状态都变为 resolved 时,新 Promise 对象的状态才会变为 resolved。
  • Promise.race():类似 Promise.all,只要有一个 Promise 对象的状态变为 resolved,新 Promise 状态就会立刻变为 resolved,只有当所有的 Promise 对象状态都变为 rejected 时,新 Promise 对象的状态才会变为 rejected。

基本用法

new Promise((resolve, reject) => {
    if (condition) {
        resolve(data)
    } else {
        reject(err)
    }
})
.then((data) => {
    //do something
    }, (err) => {
    //do something  
    })

值与状态的传递

下面讨论一些特殊情况下值与状态的传递。

链式调用

.then() 方法和 .catch() 方法都会返回一个新的 Promise 对象,这个对象与原 Promise 对象是不同的。

如果 .then() 方法和 .catch() 方法没有对 Promise 的事件进行处理,那么新的 Promise 将会继承原 Promise 对象的 [[PromiseStatus]] 和 [[PromiseValue]]。
例如以下代码将会打印 1。第一个 then() 没有对事件进行任何处理,因此第二个 then() 可以拿到原 Promise 传出的 1。

new Promise((resolve, reject) => {
  reject(1);
})
.then(() => {})
.then(() => {},
      (n) => {console.log(n);});

如果 .then() 方法和 .catch() 方法对事件进行了处理,那么新的 Promise 对象 [[PromiseStatus]] 将会是 resolved,[[PromiseValue]] 将会是 undefined。

传入 Promise 作为参数

当 resolve() 方法或者 reject() 方法传入的参数为一个 Promise 对象时会等待其中这个 Promise 对象的状态改变,然后才返回。返回的 Promise 对象的 [[PromiseStatus]] 和 [[PromiseValue]] 都与内层 Promise 相同。例如以下代码将在一秒后打印 '1 reject'。

new Promise((resolve, reject) => {
  resolve(
    new Promise((resolve, reject) => {setTimeout(() => {reject(1);}, 1000);})
  );
})
.then((e) => {console.log(e + ' resolve');},
      (e) => {console.log(e + ' reject');});

需要注意的是,上文说的“等待其中这个 Promise 对象的状态改变”并不是真的说代码会停止在 resolve 处不再执行,实际上代码会继续向下执行并且暂时返回一个状态为 pending 的 Promise 对象,等待状态的改变。例如以下代码将立即打印 2,然后再一秒后打印 '1 reject'

new Promise((resolve, reject) => {
  resolve(
    new Promise((resolve, reject) => {setTimeout(() => {reject(1);}, 1000);})
  );
      //↓
  console.log(2);
      //↑
})
.then((e) => {console.log(e + ' resolve');},
      (e) => {console.log(e + ' reject');});

resolve()方法和 reject() 方法只能够调用一次

前文说到 Promise 的状态只能够改变一次,实际上是 resolve() 方法和 reject() 方法只能够调用一次。例如:

new Promise((resolve, reject) => {
  resolve(
    new Promise(
                              //↓
        () => {setTimeout(() => {reject(1);}, 1000);}
                              //↑
    )
  );
})
.then((e) => {console.log(e + ' resolve');},
      (e) => {console.log(e + ' reject');});

这段代码不会打印任何内容,这段代码与前一段的不同之处在于箭头处调用的 reject() 方法是外层 Promise 的 reject() 方法。由于先调用了 resolve() 方法,所以这个 reject() 方法是无效的,另外,resolve() 方法又在等待内部的 Promise 改变状态。因此这个 Promise 对象会一直处于 pending 状态。

Promise.all() 和 Promise.race()

以 Promise.all() 为例,Promise.race() 反之。

resolved

只要有一个 Promise 对象调用 reject(),新 Promise 状态就会立刻变为 rejected,返回的 [[PromiseValue]] 为最先调用 reject 方法的 [[PromiseValue]]。例如以下代码将会立刻打印 [object Promise] reject,然后报错。

Promise.all([
  new Promise((resolve, reject) => {
    setTimeout(() => {reject(1);}, 1000);
  }),
  
  new Promise((resolve, reject) => {
    setTimeout(() => {resolve(2);}, 2000);
  }),
  new Promise((resolve, reject) => {
                                                            //↓
    reject(new Promise((resolve, reject) => {setTimeout(() => {reject(3)}, 5000);}));
                                                            //↑
  })
]).then((n) => {console.log(n + ' resolve');}, (n) => {console.log(n + ' reject');});

第三个 Promise 的 reject() 方法中传入的 Promise 对象的回调函数依然会被执行,但是 .all() 方法并不会等待内部 Promise 的状态改变,而是直接将返回的 Promise 对象的状态改为 rejected。而且内部 Promise 的状态也不会被传出来,而是因为没有处理而抛出错误。同理,将箭头处的 reject 改为 resolve,打印的依然会立刻打印 [object Promise] reject。再同理,以下代码依然会立刻打印 [object Promise] reject。

Promise.all([
  new Promise((resolve, reject) => {
    setTimeout(() => {reject(1);}, 1000);
  }),
  new Promise((resolve, reject) => {
    setTimeout(() => {resolve(2);}, 2000);
  }),
  new Promise((resolve, reject) => {
    reject(new Promise((resolve, reject) => {resolve(3)}));
  })
]).then((n) => {console.log(n + ' resolve');}, (n) => {console.log(n + ' reject');});

rejected

只有当内部所有的 Promise 都调用了 resolve() 方法时,返回的 Promise 才会将状态改变为 resolved,其 [[PromiseValue]] 为所有 [[PromiceValue]] 组成的数组。例如以下代码将在 5 秒后打印 [1, 2, 3]。

Promise.all([
  new Promise((resolve, reject) => {
    setTimeout(() => {resolve(1);}, 1000);
  }),
  new Promise((resolve, reject) => {
    setTimeout(() => {resolve(2);}, 2000);
  }),
  new Promise((resolve, reject) => {
    setTimeout(() => {resolve(3);}, 5000);
  })
]).then((n) => {console.log(n);}, (n) => {console.log(n);});

而对于传入 Promise 对象作为参数的情况处理方式又有不同,外部 Promise 对象会等待内部的 Promise 对象状态全部改变为 resolved,之后才会改变状态为 resolved。例如以下代码将在 5 秒后打印 '3 reject'。虽然 resolve() 方法已经被调用了,但是还是会等待它的状态改变。

Promise.all([
  new Promise((resolve, reject) => {
    setTimeout(() => {resolve(1);}, 1000);
  }),
  new Promise((resolve, reject) => {
    setTimeout(() => {resolve(2);}, 2000);
  }),
  new Promise((resolve, reject) => {
    resolve(new Promise((resolve, reject) => {setTimeout(() => {reject(3)}, 5000);}));
  })
]).then((n) => {console.log(n + ' resolve');}, (n) => {console.log(n + ' reject');});
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,686评论 6 492
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,668评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 158,160评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,736评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,847评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,043评论 1 291
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,129评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,872评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,318评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,645评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,777评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,470评论 4 333
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,126评论 3 317
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,861评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,095评论 1 267
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,589评论 2 362
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,687评论 2 351

推荐阅读更多精彩内容

  • Promiese 简单说就是一个容器,里面保存着某个未来才会结束的事件(通常是一个异步操作)的结果,语法上说,Pr...
    雨飞飞雨阅读 3,352评论 0 19
  • 00、前言Promise 是异步编程的一种解决方案,比传统的解决方案——回调函数和事件——更合理和更强大。它由社区...
    夜幕小草阅读 2,129评论 0 12
  • Promise 的含义 一句话概括一下promise的作用:可以将异步操作以同步操作的流程表达出来,避免了层层嵌套...
    雪萌萌萌阅读 5,461评论 0 7
  • 本文适用的读者 本文写给有一定Promise使用经验的人,如果你还没有使用过Promise,这篇文章可能不适合你,...
    HZ充电大喵阅读 7,299评论 6 19
  • 嘿,听说过“套路”吗? 你答:听过,很熟,想起来心里还有阵痛? 嗯嗯,咱们都懂。 如果你答:什么是套路?那是什么样...
    胖成成阅读 552评论 0 14