位置: 编程技术 - 正文

浅析Javascript ES6中的原生Promise

编辑:rootadmin

推荐整理分享浅析Javascript ES6中的原生Promise,希望有所帮助,仅作参考,欢迎阅读内容。

文章相关热门搜索词:,内容如对您有帮助,希望把文章链接给更多的朋友!

前言

一个 Promise 对象可以理解为一次将要执行的操作(常常被用于异步操作),使用了 Promise 对象之后可以用一种链式调用的方式来组织代码,让代码更加直观。而且由于 Promise.all 这样的方法存在,可以让同时执行多个操作变得简单。

Promise的兴起,是因为异步方法调用中,往往会出现回调函数一环扣一环的情况。这种情况导致了回调金字塔问题的出现。不仅代码写起来费劲又不美观,而且问题复杂的时候,阅读代码的人也难以理解。

举例如下:

假设有一个数据库保存操作,一次请求需要在三个表中保存三次数据。那么我们的代码就跟上面的代码相似了。这时候假设在第二个db.save出了问题怎么办?基于这个考虑,我们又需要在每一层回调中使用类似try...catch这样的逻辑。这个就是万恶的来源,也是node刚开始广为诟病的一点。

另外一个缺点就是,假设我们的三次保存之间并没有前后依赖关系,我们仍然需要等待前面的函数执行完毕, 才能执行下一步,而无法三个保存并行,之后返回一个三个保存过后需要的结果。(或者说实现起来需要技巧)

不幸的是,在我刚开始接触node的时候,我写了大量这样的hell。

后来因为还是写前端代码多一些,我接触了ES6,发现了一个解决回调深渊的利器Promise。

其实早在ES6的Promise之前,Q,when.js,bluebird等等库早就根据Promise标准(参考Promise/A+)造出了自己的promise轮子。 (看过一篇文章,我觉得很有道理。里面说,不要扩展内置的原生对象。这种做法是不能面向未来的。所以这里有一个提示:使用扩展原生Promise的库时,需要谨慎。)

这里仅讨论原生的Promise。

ES6 Promise

Promise对象状态

在详解Promise之前,先来点理论:

Promise/A+规范, 规定Promise对象是一个有限状态机。

它三个状态:

1、pending(执行中)

2、fulfilled(成功)

3、reject(拒绝)

其中pending为初始状态,fulfilled和rejected为结束状态(结束状态表示promise的生命周期已结束)。

状态转换关系为:

随着状态的转换将触发各种事件(如执行成功事件、执行失败事件等)。

Promise形式

Promise的长相就像这样子:

这里的变量promise是Promise这个对象的实例。

promise对象在创建的时候会执行func函数中的逻辑。

浅析Javascript ES6中的原生Promise

逻辑处理完毕并且没有错误时,resolve这个回调会将值传递到一个特殊的地方。这个特殊的地方在哪呢?就是下面代码中的then,我们使用then中的回调函数来处理resolve后的结果。比如上面的代码中,我们将值简单的输出到控制台。如果有错误,则reject到then的第二个回调函数中,对错误进行处理。

配合上面的有限状态机的理论,我们知道在Promise构造函数中执行回调函数代码时,状态为pending,resolve之后状态为fulfilled,reject之后状态为reject

Promise数据流动

以上是promise的第一次数据流动情况。

比较funny的是,promise的then方法依然能够返回一个Promise对象,这样我们就又能用下一个then来做一样的处理。

第一个then中的两个回调函数决定第一个then返回的是一个什么样的Promise对象。

假设第一个then的第一个回调没有返回一个Promise对象,那么第二个then的调用者还是原来的Promise对象,只不过其resolve的值变成了第一个then中第一个回调函数的返回值。

假设第一个then的第一个回调函数返回了一个Promise对象,那么第二个then的调用者变成了这个新的Promise对象,第二个then等待这个新的Promise对象resolve或者reject之后执行回调。

话虽然饶了一点,但是我自我感觉说的还是很清楚的呢。哈哈~

如果任意地方遇到了错误,则错误之后交给遇到的第一个带第二个回调函数的then的第二个回调函数来处理。可以理解为错误一直向后reject, 直到被处理为止。

另外,Promise对象还有一个方法catch,这个方法接受一个回调函数来处理错误。

即:

假设对错误的处理是相似的,这个方法可以对错误进行集中统一处理。所以其他的then方法就不需要第二个回调啦~

控制并发的Promise

Promise有一个"静态方法"——Promise.all(注意并非是promise.prototype), 这个方法接受一个元素是Promise对象的数组。

这个方法也返回一个Promise对象,如果数组中所有的Promise对象都resolve了,那么这些resolve的值将作为一个数组作为Promise.all这个方法的返回值的(Promise对象)的resolve值,之后可以被then方法处理。如果数组中任意的Promise被reject,那么该reject的值就是Promise.all方法的返回值的reject值.

很op的一点是:

then方法的第一个回调函数接收的resolve值(如上所述,是一个数组)的顺序和Promise.all中参数数组的顺序一致,而不是按时间顺序排序。

还有一个和Promise.all相类似的方法Promise.race,它同样接收一个数组,只不过它只接受第一个被resolve的值。

将其他对象变为Promise对象

Promise.resovle方法,可以将不是Promise对象作为参数,返回一个Promise对象。

有两种情形:

假设传入的参数没有一个.then方法,那么这个返回的Promise对象变成了resolve状态,其resolve的值就是这个对象本身。

假设传入的参数带有一个then方法(称为thenable对象), 那么将这个对象的类型变为Promise,其then方法变成Promise.prototype.then方法。

Promise是解决异步的方案吗?

最后说一点很重要的事:Promise的作用是解决回调金字塔的问题,对于控制异步流程实际上没有起到很大的作用。真正使用Promise对异步流程进行控制,我们还要借助ES6 generator函数。(例如Tj大神的co库的实现)。

然而ES7将有一个更加牛逼的解决方案:async/await,这个方案类似于co,但是加了原生支持。拭目以待吧。

总结

标签: 浅析Javascript ES6中的原生Promise

本文链接地址:https://www.jiuchutong.com/biancheng/373577.html 转载请保留说明!

上一篇:详解Javascript ES6中的箭头函数(Arrow Functions)

下一篇:js前端面试题及答案整理(一)(前端面试题csdn)

  • 印花税不足1元需要缴纳吗
  • 报税扣款锁定如何解除
  • 城镇土地使用税的计算公式
  • 已交增值税如何做账
  • 个税负数怎么做账
  • 设计服务类合同
  • 固定资产替换公式
  • 税收的构成要素包含哪些内容
  • 购买非专利技术发生的手续费计入什么科目
  • 丢失发票联和抵扣联
  • 农产品进项抵扣及申报表填写案例
  • 商场收取租户电费怎么处理税收?
  • 企业丢失账簿稽查局可以处罚吗
  • 增值税为什么不重复征税
  • 专利技术评估价值入股价偏高说明什么
  • 银行贷款利息支出汇算清缴需要调整吗
  • 企业长期待摊费用包括
  • 怎么通过合并弥补亏损,使得利益最大化?
  • 购买防疫物资计划
  • 吸收合并后被合并的公司
  • 日常管理的基本内容及方法是什么
  • 出租房产免收租金合法吗
  • 分公司使用总公司业绩投标
  • 小企业执行新会计准则吗
  • 税款申报成功但没有缴款怎么办
  • 如何找回windows密钥
  • php到底是什么
  • msp 什么意思
  • 补缴税金会计分录
  • 企业向股东发放现金股利会引起
  • php的循环
  • 库存现金清查主要包括哪些内容
  • Element-UI--<el-switch>的@change回调函数的参数用法
  • 独立核算分公司和非独立核算分公司
  • Yii CFileCache 获取不到值的原因分析
  • 个人所得税的现状分析
  • php二维数组的遍历
  • python爬虫案例题目
  • 开具销项负数发票 矿产品什么意思
  • 物业管理公司经营范围有哪些内容
  • 个人所得税既有工资薪金又有劳务报酬房屋租金
  • 节能控制模块
  • 增值税税款抵扣
  • 无形资产一般包括哪些权利
  • mysql 索引 key
  • 预付房租计入长期待摊费用吗
  • 租赁物品损坏赔偿金额标准
  • 给客户送礼计入成本吗
  • 1000元的打印机双十一满减可以减150吗少
  • 房产税的计算方式
  • 工资走公账有什么好处
  • 建筑劳务公司的会计账务处理
  • 纳税申报表多缴了怎么办
  • 购置一项设备,有两种可供选择
  • 无发票的费用怎么算
  • 收到融资租赁发票要交印花税吗
  • 在建工程转固定资产的账务处理
  • 私营公司的钱怎么拿出来
  • 怎样计算存款利息?
  • 什么是逐步结转分步法
  • mysql免安装版本
  • centos7 syslog
  • Windows设置启动引导
  • 流言:Win9或终止32位系统 加速64位平台
  • 详细介绍的英文
  • ssh permission denied password
  • win7 64位旗舰版设置插上耳机就能播放声音拔下耳机就自动禁音方法
  • win8分屏快捷键
  • 如何深度理解
  • cocos2dx4.0教程
  • java框架怎么用
  • sku分析的作用
  • unity 2Dtoolkit 插件创建中文字体
  • jquery.validate[.unobtrusive]和Bootstrap实现tooltip错误提示问题分析
  • node.js deno
  • 天津武清房管局电话
  • 江苏税务开发票
  • 购进货物计入什么科目
  • 江苏国税,地税怎么交
  • 陕西税务干部学校副教授李刚简历
  • 免责声明:网站部分图片文字素材来源于网络,如有侵权,请及时告知,我们会第一时间删除,谢谢! 邮箱:opceo@qq.com

    鄂ICP备2023003026号

    网站地图: 企业信息 工商信息 财税知识 网络常识 编程技术

    友情链接: 武汉网站建设