关于Promise

一、为何会有Promise?

在JavaScript的世界中,所有代码都是单线程执行的。由于这个“缺陷”,导致JavaScript的所有网络操作,浏览器事件,都必须是异步执行。在Promise出现之前是通过回调函数(callback)实现异步, 简单说回调函数就是将一个方法func2作为参数传入另一个方法func1中,当func1执行到某一步或者满足某种条件的时候才执行传入的参数func2。

一般来说我们会碰到的回调嵌套都不会很多,一般就一到两级,但是某些情况下,回调嵌套很多时,代码就会非常繁琐,逻辑已经很难理清楚了,这种情况俗称——回调地狱。

多层嵌套的回调中,有同步/异步的方法,那么执行顺序会变得混乱,而Promise采用链式调用,使我们的代码更容易理解和维护,而且Promise还增加了许多有用的特性,让我们处理异步编程得心应手。如下面这个例子:

//回调函数写法:
step1(function (value1) { 
  step2(value1, function(value2) { 
    step3(value2, function(value3) { 
      step4(value3, function(value4) { 
        // ...
        });
     }); 
  });
});

//Promise的写法:
(new Promise(step1))
  .then(step2)
  .then(step3)
  .then(step4);

二、什么是Promise?

  • Promise由社区最早提出和实现,ES6将其写进了语言标准,统一了语法,原生提供了Promise。
  • Promise是抽象异步处理对象以及对其进行各种操作的组件。Promise就是一个容器,里面保存着某个未来才会结束的事件(通常是一个异步操作)的结果。
  • Promise就如它的意思“承诺”一样,既然执行了,就不管程序运行结果成功还是失败,最终都会给一个答复,而这个答复就是一个Promise对象。执行流程如下图:

可能上面的阐述还是不够直观易懂,那么可参考知乎上的用“定外卖”来举的这个例子:十五行代码带你搞懂Promise

Promise对象有三种状态:
1、异步操作"未完成"(pending)
2、异步操作"已完成" (resolved)
3、异步操作"失败" (rejected)

这三种状态的变化途径:
1、异步操作从"未完成"到"已完成"
2、异步操作从"未完成“到"失败"

Promise对象的最终结果:
1、异步操作成功 Promise对象传回一个值,状态变为resolved
2、异步操作失败 Promise对象抛出一个错误,状态变为rejected
注意:有且只有这2种结果,并且状态一旦改变,就无法再次改变状态,这也是它名字“承诺”的由来。

三、基本用法

1、先声明一个Promise对象,生成Promise实例。

let p = new Promise((resolve, reject) => {
  // ...

  if (/* 异步操作成功 */){
    resolve(value);//成功抛出value
  } else {
    reject(error);//失败抛出错误
  }
});

2、Promise实例生成以后,可以用then方法分别指定resolved状态和reject状态的回调函数。

p.then((value) => {
  // 成功执行
}, (error) => {
  // 失败执行   --->可选
});

注意:实例化的Promise对象会立即执行

现在我们可以开始写一个简单的例子,比如随机生成一个数,1秒后判断可不可以做除数(0不能做除数),代码如下:

let p = new Promise((resolve, reject) => {
    var divisor = Math.random();
    setTimeout(() => {
        if(divisor) {
            resolve('可以做除数~');
        }
        else {
            reject('不可以做除数~');
        }
    }, 1000);
});

console.log(p); // 在浏览器的控制台运行的话,它返回的是一个包含了许多属性的Promise对象

p.then((result) => {
    console.log(result);
}, (err) => {
    console.log(err);
}); // 1s后这里的输出可能是fail也可能是success

四、常见方法

1、Promise.then()

promise.then(
    () => { console.log('this is success callback') },
    () => { console.log('this is fail callback') }
)

.then()方法是Promise原型链上的方法,它包含两个参数方法,分别是已成功resolved的回调和已失败rejected的回调。

2、Promise.catch()

promise.then(
    () => { console.log('this is success callback') }
).catch(
    (err) => { console.log(err) }
)

.catch()的作用是捕获Promise的错误,与then()的rejected回调作用几乎一致。但是由于Promise的抛错具有冒泡性质,能够不断传递,这样就能够在下一个catch()中统一处理这些错误。
同时catch()也能够捕获then()中抛出的错误,所以建议不要使用then()的rejected回调,而是统一使用catch()来处理错误。
同样,catch()中也可以抛出错误,由于抛出的错误会在下一个catch中被捕获处理,因此可以再添加catch()。

3、Promise.all()

var promise = Promise.all( [p1, p2, p3] )
promise.then(
   //  ...
).catch(
   // ...
)

当p1、p2、p3的状态都变成resolved时,promise才会变成resolved,并调用then()的已完成回调,但只要有一个变成rejected状态,promise就会立刻变成rejected状态。

4、Promise.race()

var promise = Promise.race( [p1, p2, p3] )
promise.then(
   // ...
).catch(
   // ...
)

race()方法,参数与Promise.all()相同,不同的是,参数中的p1、p2、p3只要有一个改变状态,promise就会立刻变成相同的状态并执行对于的回调。

5、Promise.done()
Promise.done()的用法类似.then() ,可以提供resolved和rejected方法,也可以不提供任何参数,它的主要作用是在回调链的尾端捕捉前面没有被.catch() 捕捉到的错误。

6、Promise.finally()
Promise. finally()接受一个方法作为参数,这个方法不管promise最终的状态是怎样,都一定会被执行。

五、缺点

说了这么多Promise的好用之处,那是不是堪称完美的呢?不是的,下面列举了它的几条缺点,也是我们在使用过程中需要注意的地方。

  • 无法取消Promise,一旦新建它就会立即执行,无法中途取消。
  • 如果不设置回调函数,Promise内部抛出的错误,不会反应到外部。
  • 当处于Pending状态时,无法得知目前进展到哪一个阶段(刚刚开始还是即将完成)。

六、演变(async/await)

Promise让我们告别回调函数,写出更优雅的异步代码;在实践过程中,却发现Promise并不完美;技术进步是无止境的,这时,便有了async/await。
使用async/await,搭配promise,可以通过编写形似同步的代码来处理异步流程,提高代码的简洁性和可读性。

async
使用async function可以定义一个异步函数,不管在函数体内return了什么值,async函数的实际返回值总是一个Promise对象。

await

await操作符用于等待一个Promise对象,它只能在异步函数async function内部使用。

需要注意的是,await表达式会暂停当前async function的执行,等待Promise处理完成。若Promise正常处理,其处理结果作为await表达式的值,继续执行async function;若Promise处理异常,await表达式会把Promise的异常原因抛出。

相信大家都已经在项目中使用过async/await语法了,那么我就不介绍它的用法了,直接说说需要注意哪些地方。

  • 避免直接将await调用当作变量,例如:
// 错误写法
initData(await getData());
// 正确写法
const data = await getData();
initData(data);

// 错误写法
const obj = {
  data: await getData()
}
// 正确写法
const data = await getData();
const obj = {
  data
}
  • await只能用于async声明的函数上下文中, 例如:
/* 场景:多个专题获取并处理报告内容 */

// 错误写法
async handleInited() {
    this.reportData.map(v => {
        const reportData = await this.handleReport(v.topicId);
        return reportData;
     });
},
// 正确写法
handleInited() {
    this.reportData.map(async v => { // 从语法上来说,给map的callback加上async才可以运行
        const reportData = await this.handleReport(v.topicId);
        return reportData;
     });
},
  • 注意异步操作的依赖关系,避免滥用async/await,例如:
/* 场景:多个专题获取并处理报告内容 */

// 错误写法
handleInited() {
    this.reportData.map(async v => {
        const reportData = await this.handleReport(v.topicId);
        return reportData;
     });
     this.showLoading = false; // 并不会等待上面代码返回结果后再执行
},
// 正确写法
async handleInited() {
    Promise.all(this.reportData.map(v => {
        return this.handleReport(v.topicId);
     })).then(() => {
        this.showLoading = false;
    });
},

从上面的代码可以看出,我们是想要handleReport函数并行执行,等到都返回结果后再将showLoading置为false。如果使用async/await,显然不能达到我们想要的,await只会暂停mapcallback,因此map完成时,不能保证handleReport也全部完成,这时使用之前提到的Promise.all()再合适不过了。

七、总结

JavaScript的异步编写方式,从回调函数到Promise再到async/await,表面上只是写法的变化,本质上则是语言层的一次次抽象,让我们可以用更简单的方式实现同样的功能,而不需要去考虑代码是如何执行的。

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

推荐阅读更多精彩内容

  • Promise 对象 Promise 的含义 Promise 是异步编程的一种解决方案,比传统的解决方案——回调函...
    neromous阅读 8,704评论 1 56
  • Promise对象是一种解决异步问题的方法,还有的解决方案是asyns 和 await (es7) 这么是目前的终...
    站在大神的肩膀上看世界阅读 1,262评论 0 6
  • 说到异步,怎么说还是得有Promise这玩意 Promise 的含义 Promise 是一个保存着某个未来才会结束...
    LElysion阅读 672评论 0 0
  • 目录:Promise 的含义基本用法Promise.prototype.then()Promise.prototy...
    BluesCurry阅读 1,492评论 0 8
  • 执简书之手,尽余生之力,涤尘土之垢,视人间之欲,思入世之秘,留真情给你。你若来过,我变记得。努力更文于活着的每一天!
    绮灵儿阅读 347评论 6 4