资讯专栏INFORMATION COLUMN

ES6中的异步编程:Generators函数(一)

ztyzz / 2303人阅读

摘要:由于可以使用语句来暂停异步操作,这让异步编程的代码,很像同步数据流方法一样。该临时函数就叫做函数。下面就是简单的函数转换器。

访问原文地址

对ES6的generators的介绍分为3个部分

第一部分base介绍及使用

第二部分基于generators和Promise实现最强大的异步处理逻辑

概述

Generator函数是协程在ES6的实现,用来做异步流程的封装,最大特点就是可以交出函数的执行权(即暂停执行)。十分的奇葩,光看语法,简直认不出这也是JavaScript了。由于可以使用yield语句来暂停异步操作,这让generators异步编程的代码,很像同步数据流方法一样。因为从语法角度来看,generators函数是一个状态机,封装了多个内部状态,通过iterator来分步调用。

基本语法 2个关键字搞定generators语法

function与函数名直接的星号:*

函数体内yield语句

function* testGenerator() {
    yield "first yield";
    yield "second yield";
    return "last";
}

var gen = testGenerator();

console.log(gen.next().value);// first yield 
// { value: "first yield", done: false }
console.log(gen.next().value);// second yield
// { value: "second yield", done: false }
console.log(gen.next().value);// last 
// { value: "last", done: true }

console.log(gen.next().value);// undefined
for...of遍历

for...of循环可以自动遍历generators函数的iterator对象,且不再需要调用next方法。for...of需要检查iterator对象的done属性,如果为true,则结束循环,因此return语句不能被遍历到

for (let i of testGenerator) {
    console.log(i);
}
// first yield
// second yield
next方法的参数

yield句本身没有返回值,或者说总是返回undefined。next方法可以带一个参数,该参数就会被当作上一个yield语句的返回值。

function *gen(){
  let arr = [];
  while(true){
    arr.push(yield arr);
  }
}

var name = gen();

console.log(name.next("first").value);//[]
console.log(name.next("second").value);//["second"]
console.log(name.next("thrid").value);//["second","thrid"]

需要注意的是,第一次执行next设置参数没有效果。

generators实践 实现Fibonacci数列

递归实现:

function* fib (n, current = 0, next = 1) {
  if (n === 0) {
    return 0;
  }

  yield current;
  yield* fib(n - 1, next, current + next);
}

for (let n of fibonacci()) {
  if (n > 1000) break;
  console.log(n);
}

注:如果存储计算结果再过运算,这样的实现比递归方法效率高3倍

function* fibonacci() {
  let [prev, curr] = [0, 1];
  for (;;) {
    [prev, curr] = [curr, prev + curr];
    yield curr;
  }
}

for (let n of fibonacci()) {
  if (n > 1000) break;
  console.log(n);
}
利用for...of循环,遍历任意对象(object)的方法

原生的JavaScript对象没有遍历接口,无法使用for...of循环,通过Generator函数为它加上这个接口,就可以用了。

function* objectEntries(obj) {
  let propKeys = Reflect.ownKeys(obj);

  for (let propKey of propKeys) {
    yield [propKey, obj[propKey]];
  }
}

let jane = { first: "Jane", last: "Doe" };

for (let [key, value] of objectEntries(jane)) {
  console.log(`${key}: ${value}`);
}
// first: Jane
// last: Doe
ES6中iterator遍历接口汇总

for...of循环

扩展运算符(...)

解构赋值

Array.from方法内部调用的

它们都可以将Generator函数返回的Iterator对象,作为参数来使用。

function* numbers () {
  yield 1
  yield 2
  return 3
}

// 扩展运算符
[...numbers()] // [1, 2]

// Array.from 方法
Array.from(numbers()) // [1, 2]

// 解构赋值
let [x, y] = numbers();
x // 1
y // 2

// for...of 循环
for (let n of numbers()) {
  console.log(n)
}
// 1
// 2
generators与同步

generators一个特点就是代码看上去非常像同步编程的效果

function* test() {
    yield( "1st" );
    yield( "2nd" );
    yield( "3rd" );
    yield( "4th" );
}
var iterator = test();

console.log( "== Start of Line ==" );
console.log( iterator.next().value );
console.log( iterator.next().value );
for ( var line of iterator ) {
    console.log( line );
}
console.log( "== End of Line ==" );

看下输出,浓浓的同步执行风格。

== Start of Line ==
1st
2nd
3rd
4th
== End of Line ==
callback、Promises、Generators比较

举例说一个场景,查询一篇新闻文章的作者信息,流程是:请求最新文章列表->请求某文章相关id->作者id信息

callback实现
getArticleList(function(articles){
    getArticle(articles[0].id, function(article){
        getAuthor(article.authorId, function(author){
            alert(author.email);
        })
    })
})

function getAuthor(id, callback){
    $.ajax(url,{
        author: id
    }).done(function(result){
        callback(result);
    })
}

function getArticle(id, callback){
    $.ajax(url,{
        id: id
    }).done(function(result){
        callback(result);
    })
}

function getArticleList(callback){
    $.ajax(url)
    .done(function(result){
        callback(result);
    });
}
用Promise来做
getArticleList()
.then(articles => getArticle(articles[0].id))
.then(article => getAuthor(article.authorId))
.then(author => {
    alert(author.email);
});

function getAuthor(id){
    return new Promise(function(resolve, reject){
        $.ajax({
            url: id+"author.json",
            success: function(data) {
              resolve(data);
          }
        })
    });
}

function getArticle(id){
    return new Promise(function(resolve, reject){
        $.ajax({
            url: id+".json",
            success: function(data) {
              resolve(data);
          }
        })
    });
}

function getArticleList(){
    return new Promise(function(resolve, reject){
       $.ajax({
           url: "all.json",
           success: function(data) {
             resolve(data);
         }
       }) 
    });
}
Gererator来实现
function* run(){
  var articles = yield getArticleList();
  var article = yield getArticle(articles[0].id);
  var author = yield getAuthor(article.authorId);
  alert(author.email);  
}

var gen = run();
gen.next().value.then(function(r1){
  gen.next(r1).value.then(function(r2){
      gen.next(r2).value.then(function(r3){
        gen.next(r3);
        console.log("done");
      })
  })
});
runGenerator的实现

每次都要手动去调用next方法,还是会让代码变得冗长,我们可以设计一个专门用来运行generators的方法,并可以抽象出来,以后就可以做一个统一的error管理,或者获取本地数据逻辑的变化。

Thunk函数方法

编译器的‘传名调用’实现,将所有的参数放到一个临时函数中,再将这个临时函数作为参数传入到函数体中。该临时函数就叫做Thunk函数。

任何函数,只要参数有回调函数,就能写成Thunk函数的方法。下面就是简单的Thunk函数转换器。

//es5
var Thunk = function(fn) {
    return function() {
        var args = Array.pototype.silce.call(argumnets);
        return function (callback) {
            args.push(callback);
            return fn.apply(this. args);
        }
    }
}

//es6
var Thunk = function(fn) {
    return function(...args) {
        return function(callback) {
            return fn.call(this, ...args, callback);
        }
    }
}

一个使用Thunk方法来实现readFile的例子

//正常版本的readFile(多参数)
fs.readFile(filename, callback);

//Thunk版本的readFile(单参数)
var readFileThunk = Thunk(filename);
readFileThunk(callback);

var Thunk = function(fileName) {
    return function(callback) {
        return fs.readFile(fileName, callback);
    }
}

可以看到,如果我们通过构建一个基于Thunk方法实现的runGenerators函数,可以很好的控制我们的generators运行流程。

function *generator() {
    var articles = yield getArticleList();
    var article = yield getArticle(articles[0].id);
    var author = yield getAuthor(article.authorId);
    console.log(author.email);
}

function runGenerator() {
    var gen = generator();
    
    function go(result) {
        if(result.done) return;
        
        result.value.then(function(rsp) {
            go(gen.next(rsp));
        })
    }
    
    go(gen.next());
}

runGenerator();
参考

[Javascript] Promise, generator, async與ES6

Generator 函数

文章版权归作者所有,未经允许请勿转载,若此文章存在违规行为,您可以联系管理员删除。

转载请注明本文地址:https://www.ucloud.cn/yun/80914.html

相关文章

  • ES6中的异步编程Generators函数+Promise:最强大的异步处理方式

    摘要:更好的异步编程上面的方法可以适用于那些比较简单的异步工作流程。小结的组合目前是最强大,也是最优雅的异步流程管理编程方式。 访问原文地址 generators主要作用就是提供了一种,单线程的,很像同步方法的编程风格,方便你把异步实现的那些细节藏在别处。这让我们可以用一种很自然的方式书写我们代码中的流程和状态逻辑,不再需要去遵循那些奇怪的异步编程风格。 换句话说,通过将我们generato...

    Taonce 评论0 收藏0
  • JavaScript 异步编程的四种方式

    摘要:异步编程是每个使用编程的人都会遇到的问题,无论是前端的请求,或是的各种异步。本文就来总结一下常见的四种处理异步编程的方法。利用一种链式调用的方法来组织异步代码,可以将原来以回调函数形式调用的代码改为链式调用。 异步编程是每个使用 JavaScript 编程的人都会遇到的问题,无论是前端的 ajax 请求,或是 node 的各种异步 API。本文就来总结一下常见的四种处理异步编程的方法。...

    microelec 评论0 收藏0
  • ES6 异步编程:Generator

    摘要:生成器是原生提供的异步编程方案,其语法行为和传统函数完全不同,阮大的入门一书中对生成器有比较详尽的介绍,还有一些其他的文章可以参考,比如入门深入浅出三生成器深入浅出十一生成器,续篇本文主要是通过一些代码示例来记录和总结生成器的用法。 Generator 生成器是es6原生提供的异步编程方案,其语法行为和传统函数完全不同,阮大的《ECMAScript 6 入门》一书中对生成器有比较详尽的...

    Eidesen 评论0 收藏0
  • 如何理解 koa 中间件执行机制

    摘要:注是先前版本处理异步函数的方式,通过可以将异步函数封装成,传入普通参数后形成仅需要参数的偏函数,以此简化调用代码目前中的偏函数已经被无情地化了。 前几天研究了TJ的koa/co4.x和一系列koa依赖的源码,在知乎上做出了人生首次回答(而且我真得再也不想去知乎回答技术问题了_(:з」∠)_),因此把文字搬到这里。 ES2015 Generator/Yield 关于Generator...

    charles_paul 评论0 收藏0
  • 通过ES6 Generator函数实现异步流程

    摘要:换句话说,我们很好的对代码的功能关注点进行了分离通过将使用消费值得地方函数中的逻辑和通过异步流程来获取值迭代器的方法进行了有效的分离。但是现在我们通过来管理代码的异步流程部分,我们解决了回调函数所带来的反转控制等问题。 本文翻译自 Going Async With ES6 Generators 由于个人能力知识有限,翻译过程中难免有纰漏和错误,还望指正Issue ES6 Gener...

    刘厚水 评论0 收藏0

发表评论

0条评论

ztyzz

|高级讲师

TA的文章

阅读更多
最新活动
阅读需要支付1元查看
<