Promise 的使用

使用Promise

基本用法

A Promise is an object representing the eventual completion or failure of an asynchronous operation. Promise是表示异步操作最终完成或失败的对象。

Promise 本质上是一个绑定了回调函数的对象。不同于我们平时看到的将回调作为参数传递给一个函数的形式。

首先来看下,我们熟悉的将回调作为函数传递给函数的例子:

    function successCallback(result) {
      console.log("Audio file ready at URL: " + result);
    }
    
    function failureCallback(error) {
      console.log("Error generating audio file: " + error);
    }
    
    createAudioFileAsync(audioSettings, successCallback, failureCallback);

上面这个函数作用是异步的创建一个音频文件,第一个参数是配置对象,第二和第三个分别是创建音频文件成功和失败后的回调函数。

那如果将上面的例子用Promise来处理,会是怎样的形式呢?

  1. Promise用then来表示处理结果

     createAudioFileAsync(audioSettings).then(successfulCallback, failureCallback)
    
  2. 也可以写成另外一种简化形式

     createAudioFileAsync(audioSettings).then(successfulCallback).catch(failureCallback)
    

我个人喜欢第二种形式,可读性更强一些

Chain 链式调用

这个是Promise一个特别有意思的用法,对于处理多个串行的异步调用非常好用。

考虑这样的场景,我们要处理多个异步调用,每一个异步的调用都要利用前一个的结果。那么就可以使用promise的chain。

例如,我们要获取某些资源,并且对资源处理之后,发送PUT请求更新资源的结果。而第一步是验证用户的合法性,是否具有这样操作的权限。那么用Promise就可以写成如下形式:

LoginPromise(userProfile)
    .then(resp => {
        if (resp.status == OK) return resp.authToken;
    })
    .then(token => {
       return axios.get(url, {authToken: token}); 
    })
    .then(resp => {
      let resource = prepareResource();
      return axios.post(url, {resource: resource}, header);
    })
    .catch(err => {
      console.log(err);
    })

上述的代码就会比之前嵌套的回调要优雅而且表意很多。

当然,catch之后依然还可以跟then,表示出错之后的继续处理。

改造旧的异步调用

我们可能会碰到许多已经存在的异步调用,如何使用Promise来改造他们呢,看一下下面这个例子:

setTimeout(() => saySomething("10 seconds passed"), 10000);

setTimeout 设置一个定时器,在10000ms之后发出消息。

const wait = ms => new Promise(resolve => setTimeout(resolve, ms));

wait(10000).then(() => saySomething("10 seconds")).catch(failureCallback);

可以看到wait只是作为一个Promise对象存在,至于回调要做什么,在then中可以来设置。

组合

Promise.resolve() 和 Promise.reject() 可以手动创建一个已经resolve或者reject的Promise对象。在某些场景下是一个非常有用的特性。

Promise.all() 和 Promise.race() 用来处理并行的Promise。

例如下面的代码,就使用了Promise和reduce:

[func1, func2].reduce((p, f) => p.then(f), Promise.resolve());

首先创建Promise.resolve(),然后依次处理func1和func2. 上述代码等价于:

Promise.resolve().then(func1).then(func2);

以此类推,就可以将上述代码扩展到多个函数的处理:

const applyAsync = (acc,val) => acc.then(val);
const composeAsync = (...funcs) => x => funcs.reduce(applyAsync, Promise.resolve(x));

composeAsync是一个接收可变参数作为一组回调函数的函数。使用方式如下:

const transformData = composeAsync(func1, asyncFunc1, asyncFunc2, func2);

composeAsync 的返回值是一个函数,此函数输入参数为x,执行Promise.resolve(x).then(func1).then(func2).then(func3)...

transformData(data)

这个函数实现的功能是对data,由一组函数(func1, asyncFunc1, asyncFunc2, func2)来处理。

Nesting

这里需要注意的是Promise的Nesting,嵌套。如果你想实现的是Promise的串行调用,那么就一定使用Promise的Chain来操作。

如果出现Promise的嵌套,就会出现一些隐藏的问题,看如下的例子:

promise1()
    .then(res => {
      promise2(res)
        .then(newRes => {
        })
        .catch(err) {
          console.log(err)
        }
    })
    .then(result => {
      promise3()
        .then( () => {})
        .catch(() => {})
    })
    .catch(err => {})

上述代码中,也许你想在promise2出现失败的情况下,后续的异步操作promise3不用再执行,但实际上做不到,内层的catch并不会传递到外层,所以then.promise3就会继续执行下去。

容易出的错

// Bad example! Spot 3 mistakes!

doSomething().then(function(result) {
  doSomethingElse(result) // Forgot to return promise from inner chain + unnecessary nesting
  .then(newResult => doThirdThing(newResult));
}).then(() => doFourthThing());
// Forgot to terminate chain with a catch!

上面的代码有三处错误:

  • 内层并没有return新的promise,这样就会到这外层不会等到内层结束,而直接运行doFourthThing()。
  • nesting 在前面已经分析过了
  • 没有catch,在众多浏览器中没有catch会导致promise出错

一个比较合理的promise例子如下:

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

推荐阅读更多精彩内容

  • title: promise总结 总结在前 前言 下文类似 Promise#then、Promise#resolv...
    JyLie阅读 12,243评论 1 21
  • Promise 对象 Promise 的含义 Promise 是异步编程的一种解决方案,比传统的解决方案——回调函...
    neromous阅读 8,706评论 1 56
  • Prepending(进行时),Resolve(成功了),Reject(失败了),then......等 1.Pr...
    _菩提本无树_阅读 49,057评论 0 21
  • 本文适用的读者 本文写给有一定Promise使用经验的人,如果你还没有使用过Promise,这篇文章可能不适合你,...
    HZ充电大喵阅读 7,309评论 6 19
  • … 刚才在浦园跑步 夜色浓重 迎面过来的一个小男孩 指着我大喊 妈妈!你快看,这个男人有一身肌肉 … 小小年纪 怎...
    AI大叔阅读 215评论 1 0