Promise网络请求,异步编程封装,优雅

1.什么是Promise,Promise到底是做什么的,有什么作用?

Promise(承诺)是ES6中一个非常重要和好用的特性,Promise是针对异步编程的一种实现解决方案,Promise内部可以用来处理异步事件和操作,即可以用于包裹(wrapper into)封装异步操作代码。

2.常见的应用场景。

  • 网络请求(网络请求=异步操作=setTimeout())

比如我们想封装一个网络请求的函数,因为不能立即拿到结果,所以不能像简单的同步编程一样将结果直接返回。


如果按照这样排着队来执行代码,同步编程就会带来一个严重的问题,那就是网络阻塞。

比如说,我们的用户在电商购物平台浏览进行某种操作(这些生鲜水果真不错,我想买几斤,用户在点击某个喜欢的商品,一般都会进一步了解商品的详情,这时候就会想点击了解下关于商品的详情,看看评论,才决定下不下订单)的时候,也就是说,用户的所有这些不确定性点击事件,会突然间要向我们的服务器发送一个网络请求(这时候的JS代码就会去请求有关这个商品的详情页反馈给用户),而网络请求一般情况下呢都是比较耗时的。

如果这个时候,我们的网络请求是同步(按照顺序排着队执行代码,不许插队)的话,就意味着用户面临网络阻塞的情形(哎哎!兄弟,我在咨询相关的业务办理呢,还没到你呢,到后面排队去,排队去!),这就导致用户界面上什么东西都不会显示,这时候就会出现了网页空白,因为当前的js代码执行已经被阻塞了,它要等到当前网络请求任务完成后,才轮到它。),

所以这时候呢,我们会传入另外一个函数,在数据请求成功时,在数据请求成功时,将数据通过传入的函数回调回去。

就是当我们有一个网络请求的时候,我们会直接开启一个异步的任务线程,让这个异步任务专门单独处理我们的请求(相当于我们去银行开办业务的时候,银行业务众多且杂,如果所有业务都集中在一个窗口办理的话,那就非常不现实了,因为有的人想转账,有的人想存钱投资理财(办理几几十分钟),有的人想单纯地办卡,七八分钟就可以办好的业务,那也要让他们安安静静的排队等待一整天?等到大晚上吗?

如果这时候,采取分门别类的对策,开出一个个专门属于让我们办理具体业务的窗口的话,那让用户办理起业务就方便了,反过来想,这就好比我们的异步编程有专属于我们自己的通道,办卡在1号窗口,转账在2号窗口,这样我们就避免了集中在一个窗口里面排着长长的等侯队列,而直接去1号窗口办理业务即可。)

但开辟出的1号窗口,如果人太多的话,依然是排队等候呀,没错,这就是所谓的回调地狱了。

3.Promise请求过多也会带来问题。

当我们的网络请求非常复杂时,会出现回调地狱的情形,(也就是一个回调函数里面还需要一个回调,回调函数里面又需要一个回调,呈现出多层级嵌套的回调关系。)

4.网络请求的回调地狱

我们可以考虑一下下面这种嵌套场景。

  • 我们需要通过一个url1从服务器加载一个数据data1,同时data1中包含了下一个请求的url2
  • 接着,我们需要通过data1取出url2,从服务器加载数据data2,同时data2中还包含了下一个请求的url3
  • 再然后,我们需要通过data2取出url3,从服务器加载一个数据data3,同时data3中又包含了下一个请求的url4
  • 最后发送网络请求url4,获取最终的数据data4


总结:上面的代码是可以正常运行并且能够获取到我们想要的结果,不过,这样的代码难看且不说,(当代码块足够多的时候)维护起来也是要死的,因此我们需要一种更加优雅的方式来进行这种异步操作,也就是ES6为我们提供的Promise类,可以非常优雅地进行异步操作

5.Promise的基本语法使用

使用定时器(setTimeout)来模拟异步操作

  • 以前我们使用定时器的写法(简单的异步操作)


  • 通过Promise类进行封装




  • 现在我们有一个需求:延迟1s打印4次内容,继续延迟1s打印4次内容,载来延迟1s,执行4次打印内容。


  • 有意思的是,执行上面的代码,我们发现,我们自己已经陷入了回调地狱的情形。


  • 所以我们要优雅地封装代码。


  • 上面我们已经初步构建出一个框架,下面让我们把之前的代码封装进去。


  • 这样一看,我们好像也没干什么,确实也是,所以我们需要继续new Promise()进行封装,这个过程其实就是类似处理我们的网路请求。


  • 还有一个setTimeout,意味着还要继续new Promise()进行封装。


    总结:最后的代码呈现出(链式编程)的写法,代码看起来好像是更加复杂了,其实不然,我们可以更加明了的看出Promise封装的层次感,大致上的结构就是呈现:一个Promise,一个then,然后一个Promise,一个then,依此类推下去,就形成清晰的一条链子,环环衔接,这一个链条还执行完毕后,就交给下一个链条。

  • 为了加上印象,我们可以再回过头来看下这种代码。


  • 如果发送的网络请求达到上百次,按照这种写法,估计代码一环套一环写下去,开发者到最后,可能连自己都傻傻分不清楚,这对大括号到底属于谁的,这时候,维护起来就相当困扰。

6.OK,通过上面的例子,我们已经清晰认识到Promise的基本使用封装写法,那什么情况会用到Promise呢?

  • 一般情况下,当我们有异步操作时,就可以使用Promise对这个异步操作进行封装,通过new 一个Promise类。

  • 源码是这样解释的。



  • 在Promise网络请求中,如resolve成功回调则交给then处理,如reject失败回调则交给catch处理。


7.Promise的三种状态

当我们在开发中有异步操作时,就可以给异步操作包装一个Promise网络请求,通过异步操作之后会出现三种状态

  • pending: 等待状态,比如我们正在进行网络请求,或者定时器还在进行,没有到终止时间。
  • fulfill: 满足状态,当我们主动回调了resolve时,代表着请求成功了,就处于该状态,并且会回调.then()函数
  • reject: 即拒绝状态,当我们主动回调了reject的时候,也就意味着网络请求失败了,就处于该状态,并且会回调.catch()函数。

图解:

8.Promise的异步处理形式。

一种是.then()和.catch()结合形式


另一种直接在.then()回调函数里面操作,即.then(函数1, 函数2)

9.Promise的链式调用标准。

其中一种我们已经在上面的例子实现了(用链式编程解决回调地狱的情形),下面我们看下Promise另一种的链式调用标准。

需求:

  • 我们有个一个网络请求,请求到数据a后,然后自己处理10代码。
  • 对数据a做拼接处理,让他变成数据ab,然后它自己处理10代码。
  • 对原有的ab数据继续做拼接处理,让其变成数据abc,然后也让他处理相关代码。

代码原型:



  • 优雅简化版,直接通过Promise.resolve(返回结果)。


  • 至简优雅版,直接return返回结果,省略掉Promise.resolve(),因为Promise内部有对应的API接口支持,会自动处理我们的逻辑绕绕。

    总结:上面的例子就是这种链式调用方式的层层演进,大道至简。

我们上面考虑到的都是网络请求成功后拿到数据的情形,那有没有可能在某一层的网路请求被拒绝pass掉了,导致后面层拿不到数据呢?这时候我们该怎么办呢?


除了reject()和.catch()搭配使用处理异常外,throw 也可以直接抛出异常。

10.Promise.all()方法的使用,在某次开发需求中,可能会遇到需要发送2次网络请求的情形,这时候就会用到Promise.all()方法。

  • 上面我们做到的都是分层请求关系(层次感很清晰),但可能有时候需要并列的Promise请求.

先来看下Ajax是如何操作的。



ES6中Promise.all()方法如何操作的。



  • 到这里,我们基本了解了Promise.all()方法的使用原理,回到上面的需求实现。


  • 上面的代码是在实际开发会遇到的,但我们没有url地址和安装jQuery,自然无法使用ajax请求数据,只能当伪代码处理,无法测试出实际效果,所以我们可用setTimeout来代替ajax异步操作。



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