说说还在stage3的async/await

这篇文章介绍了ES7中 async/await,generator,co,bluebird等异步编程的关联和发展变化,值得参考。
原文 https://cnodejs.org/topic/5709cb2f94b38dcb3c09a7ac


async/await

提议 https://tc39.github.io/ecmascript-asyncawait/目前在 stage3, 没赶在 ES2016即ES7的deadline之前达到stage4, 所以只能赶下波 ES2017 了。已经用上babel的可以不用往下看了~不是你们的菜~

async function foo(){
  let b = await bar();
  return b + 1;
}
  • 调用 bar() 的返回值是一个 Promise 实例
  • foo函数中可以去等待bar的执行,类似线程join
  • 调用 foo() 的返回值是一个 Promise 实例,可以被其他 async function await

但是只要满足一点就可以了,返回值是 Promise类型,即可称之为 aysnc function. 例如

function bar(){
  return Promise.resolve(1);
}

// or
async function bar(){
  return 1;
}

这两种形式对于使用者 foo() 来说没有任何不同。

desugaring

see https://tc39.github.io/ecmascript-asyncawait/#desugaring在async/await 语法糖之下是什么呢

async function <name>?<argumentlist><body>
=>
function <name>?<argumentlist>{ return spawn(function*() <body>, this); }

是有一个 spawn 函数来帮你运行一个 generator, 并返回一个 spawn 的返回值。proposal里给出了 spawn 的实现

function spawn(genF, self) {
    return new Promise(function(resolve, reject) {
        var gen = genF.call(self);
        function step(nextF) {
            var next;
            try {
                next = nextF();
            } catch(e) {
                // finished with failure, reject the promise
                reject(e);
                return;
            }
            if(next.done) {
                // finished with success, resolve the promise
                resolve(next.value);
                return;
            }
            // not finished, chain off the yielded promise and `step` again
            Promise.resolve(next.value).then(function(v) {
                step(function() { return gen.next(v); });
            }, function(e) {
                step(function() { return gen.throw(e); });
            });
        }
        step(function() { return gen.next(undefined); });
    });
}

可以去看 co 的源码,跟 co 差不多,co要多一些将其他非promise值转为promise的过程。所以可以这样使用

function foo(){
  return co(function*(){
    let b = yield bar();
	return b + 1;
  });
}

这个 foo 跟 前面的 async function foo() { ... } 对于使用者来说没有任何区别, (foo.length 除外, 待会说).于是可以使用一个方法去生成这样一个用 generator 写的 aysnc function

// 使用 co.wrap
let foo = co.wrap(function*(){
  let b = yield bar();
  return b + 1;
});

// 使用 bluebird.coroutine
let foo = bluebird.coroutine(function*(){
  let b = yield bar();
  return b + 1;
});

所以推荐使用 co.wrap & bluebird.coroutine或者其他工具将你的函数封装成一个可以返回 Promise 的广义async function。并不需要等到 aysnc/await 变成 native. 将来迁移到async/await 只需要修改这个定义的地方即可(如bar), 调用者(如foo)并不需要知晓你的改变。

差异

async funtion foo1(n){
  let b = await bar();
  return b + n;
}

let foo2 = co.wrap(function*(n){
  let b = yield bar();
  return b + n;
});

这两种形式,抛开一个是 function declaration, 一个是 variable declaration的差异,还有一点就是

  1. foo1.length // 1
  2. foo2.length // 0

case

https://cnodejs.org/topic/56ab1c0526d02fc6626bb383 裸奔着用 generator, 那样使用者必须使用 co, 绑死使用者是很危险的哦. see

其他使用 promise.then.then.then.then … 赶紧向这边迁移吧~

其他

个人观点,不对请指正。谢谢。


python也是这样的历程, 由 @asyncio.coroutine + yield 升级到 async/awaitsee http://www.liaoxuefeng.com/wiki/0014316089557264a6b348958f449949df42a6d3a2e542c000/00144661533005329786387b5684be385062a121e834ac7000

目前python3.5支持async和await

在JavaScript中,async/await是一种用于处理异步操作的语法糖。它可以让我们以同步的方式来编写异步代码,使得代码更加清晰和易于理解。 使用async/await,我们可以在一个函数前面加上async关键字来定义一个异步函数。在异步函数内部,我们可以使用await关键字来等待一个返回Promise对象的表达式完成。 举个例子,如果我们想要计算3个数的值,然后将结果进行输出,我们可以这样做: ```javascript async function testResult() { let first = await doubleAfter2seconds(30); let second = await doubleAfter2seconds(50); let third = await doubleAfter2seconds(30); console.log(first, second, third); } testResult(); ``` 在上面的代码中,testResult函数是一个异步函数。在函数内部,我们使用await关键字来等待每个异步操作完成,并将结果赋值给相应的变量。然后,我们使用console.log来输出这些结果。 需要注意的是,await只能在异步函数内部使用。它会暂停函数的执行,直到await后面的表达式返回的Promise对象状态变为resolved(已完成)后才会继续执行。 另外,根据引用的内容,async/await这个特性已经是stage 3的建议,也就是说它已经被广泛认可并且正式进入了JavaScript的标准化进程中。 综上所述,async/await是一种用于处理异步操作的语法糖,可以让我们以同步的方式来编写异步代码,并且使得代码更加清晰易读。<span class="em">1</span><span class="em">2</span><span class="em">3</span> #### 引用[.reference_title] - *1* *3* [JavaScript中asyncawait的基本用法](https://blog.youkuaiyun.com/weixin_44019523/article/details/113986514)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] - *2* [在JS循环中使用async/await的方法](https://download.youkuaiyun.com/download/weixin_38520258/13658555)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v93^chatsearchT3_2"}}] [.reference_item style="max-width: 50%"] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值