原生的 promise 的局限性
本文来自:https://ekyu.moe/article/limits-of-native-promise-and-async-await/
众所周知,Nodejs 已原生支持 Promise 和 async/await 关键字,异步控制已经变得更加方便。
然而,如今仍有很多人选择使用第三方的 Promise 库(如 bluebird)和类似 async/await 的实现(如 co)。这并不完全是历史原因使然,而是原生 Promise 和 async/await 仍存在着许多不足之处。本文将简单地提出一些,希望能抛砖引玉。
原生 Promise 方法匮乏
尽管原生 Promise 提供了Promise.all
和Promise.race
这类基本的方法,但在日常使用中,我们经常需要像Promise.map
、Promise.reduce
、Promise.promisify
、Promise.delay
、Promise#isPending
、Promise#cancel
、Promise#timeout
这样的方法。尽管要手动实现这些也不算难(除了Promise#cancel
),但一些第三方的 Promise 库已经实现了部分这类 feature,就暂且没有必要重复造轮子了。
17-05-31 更新
刚刚发现 Nodejs 发布了 8.0.0,Changelog 中有一行吸引了我的注意。
The new
util.promisify()
API has been added [99da8e8e02] #12442.
我查了一下文档,确实加了这个方法!好兴奋!!
原生 Promise 有性能缺陷
之前看到这个问题有提到原生 Promise 的效率问题,blubird 的作者 Petka Antonov 给出了回答,这里提一下回答中的要点。
- V8 的 Promise 是用 JavaScript 而不是 C 实现的。即便使用 C 语言实现 Promise 也不能提高多少性能,因为实际上它所做的提挈都是在与 JavaScript 对象交互。
- V8 的 Promise 实现没有像 bluebird 一样的优化,比如它会为 Promise 的 handlers 创建数组,当每个Promise都必须要创建一对数组的时候会消耗大量的内存,而实际上在 99.99% 的情况下都不会将一个 Promise 分支多次,因而对这类情况进行优化能减小内存的使用。
- 即便 V8 的 Promise 像 bluebird 一样地被优化,它仍然会受到规范的牵制。这个 benchmark 用了
new Promise
(在 bluebird 中这被视为一种 anti-pattern),因为在 ES6 中并没有别的办法来生成一个根 Promise。用new Promise
来创建一个 Promise 是极慢的。首先父函数会申请一个闭包,然后传入两个独立的闭包作为参数。算起来每生成一个 Promise 总共要申请三个闭包,然而相对于一个被优化的 Promise 而言,一个闭包都已经是个更昂贵的对象了。
兼容性
并不是所有的环境都能随随便便地升级到能支持 Promise 和 async/await 的 node 版本,尤其是 async/await 需要 Nodejs 7.6.0 以上,在我写这篇 blog 的时候 Nodejs 的 LTS 版本还是 6.10.3。而相比之下,很多第三方库都做了这个 polyfill。
async function 返回的是原生 Promise
目前没有办法向 async function 的调用返回值注入别的类型的 Promise,它返回的只能是原生的。而上文提到了原生 Promise 目前存在的很多不足,在不得不使用第三方 Promise 库的情况下,也不得不在很多地方放弃 async function。尽管,将一个原生的 Promise 转为 bluebird 的 Promise 倒也不是不可能,例如:
const Bluebird = require('bluebird');
const someTask = (async () => {})();
const bluebirdSomeTask = Bluebird.resolve(someTask);
console.log(someTask instanceof Promise); //=> true
console.log(bluebirdSomeTask instanceof Bluebird); //=> true
然而用 Bluebird.resolve
这种方法来进行转换看上去就非常 anti-pattern,也徒增了代码的复杂度。
本文采用知识共享署名-非商业性使用 4.0 国际许可协议进行许可。
本文链接:https://ekyu.moe/article/limits-of-native-promise-and-async-await/
原生的 promise 的局限性的更多相关文章
- 原生Ajax + Promise
有原生写的ajax + promise嫁接下 ;(function(root){ var LD = function(obj){ if( obj instanceof LD ) return obj; ...
- promise应用及原生实现promise模型
一.先看一个应用场景 发送一个请求获得用户id, 然后根据所获得的用户id去执行另外处理.当然这里我们完全可以使用回调,即在请求成功之后执行callback; 但是如果又添加需求呢?比如获得用户id之 ...
- 【es6】js原生的promise
JavaScript 是单线程的,这意味着任何两句代码都不能同时运行,它们得一个接一个来.在浏览器中,JavaScript 和其他任务共享一个线程,不同的浏览器略有差异,但大体上这些和 JavaScr ...
- js原生方法promise的实现
一会儿就要回家过年了,再来手写一个promise吧,要不等着下班真的煎熬... <!DOCTYPE html> <html lang="en"> <h ...
- 浅谈ES6原生Promise
浅谈ES6原生Promise 转载 作者:samchowgo 链接:https://segmentfault.com/a/1190000006708151 ES6标准出炉之前,一个幽灵,回调的幽灵,游 ...
- angular2系列教程(七)Injectable、Promise、Interface、使用服务
今天我们要讲的ng2的service这个概念,和ng1一样,service通常用于发送http请求,但其实你可以在里面封装任何你想封装的方法,有时候控制器之间的通讯也是依靠service来完成的,让我 ...
- ES6 - Note5:Promise
1.Promise介绍 Promise最早是社区提出和实现,后面ES6将其写入标准,并原生提供Promise对象,是一种异步编程的解决方案,具体的概念大家可以去查看相关的资料.传统上处理异步都是以ca ...
- js promise chain
新的标准里增加了原生的Promise. 这里只讨论链式使用的情况,思考一下其中的细节部分. 一,关于 then() 和 catch() 的复习 then() 和 catch() 的参数里可以放置 ca ...
- Promise对象
1.Promise思想:每一个异步任务立刻返回一个Promise对象,由于是立刻返回,所以可以采用同步操作的流程.这个Promises对象有一个then方法,允许指定回调函数,在异步任务完成后调用. ...
随机推荐
- Python全栈day21(作业针对一个文件进行查询修改删除的操作练习)
需求,有一个配置文件test.conf内容如下 backend www1 server 1 server 2 backend www2 server 3 server 4 add [{'backend ...
- 启动windows服务的bat文件编写格式
1.bat文件需要和bin文件内容放在一起 启动服务的bat文件如下: sc create 邮件服务 binPath= "%~dp0可执行文件名称.exe" start= auto ...
- WEB项目中使用UEditor(富文本编辑器)
Ueditor富文本编辑器是在很多项目里经常用到的框架,是百度开发团队开发的一款很好用的富文本编辑器 下面就是我在一个系统里用到的,有了富文本编辑器,管理员使用起来不是很方便? 所以本博客介绍这个富文 ...
- POJ1128 Frame Stacking(拓扑排序)
题目链接:http://poj.org/problem?id=1128 题意:给你一个平面,里面有些矩形(由字母围成),这些矩形互相有覆盖关系,请从求出最底层的矩形到最上层的矩形的序列,如果存在多种序 ...
- 删除Android自带的系统软件注意事项
教程类 知识分享 [转]好多童鞋在ROOT手机后,大刀阔斧的就开始砍系统里面的东西,有些事删不得的,删除错了就成砖头了! 以下是对照表: 注意:打*号的千万别删,打-号的是建议删的(大多要穿墙才能 ...
- golang官方实现如何对httpserver做频率限制(最大连接数限制)
一般海量处理服务,都会对服务做个最大连接数限制,超过该限制之后,拒绝服务,避免发生雪崩,压坏服务. 使用golang来编写httpserver时,如何进行呢?官方已经有实现好的包. 使用示例: imp ...
- Python程序员鲜为人知但你应该知道的16个问题(转)
add by zhj: 没找到原文出处,只能找到转载的,文中说有17个坑,其实是16个 全文如下 这篇文章主要介绍了Python程序员代码编写时应该避免的16个“坑”,也可以说成Python程序员代码 ...
- BBS项目部署之数据库的处理
一 数据库的处理: 1.1上传bbs.sql(数据库中的数据) 1.2在mysql中创建bbs库,并导入数据库SQL脚本 在mysqld的文件夹中: mysql> create databas ...
- Spark2.0机器学习系列之10: 聚类(高斯混合模型 GMM)
在Spark2.0版本中(不是基于RDD API的MLlib),共有四种聚类方法: (1)K-means (2)Latent Dirichlet allocation (LDA) ...
- math.h函数库
C语言中之数学函数 C语言提供了以下的数学函数,要使用这些函数时,在程序文件头必须加入: #include <math.h> 编译时,必须加上参数「-lm」(表示连结至数学函式库),例如「 ...