es6解读7- Generator

作者: cd72c1240b33 | 来源:发表于2017-07-17 23:39 被阅读133次

    Generator

    基本概念

    Generator函数有多种理解角度。从语法上,首先可以把它理解成,Generator函数是一个状态机,封装了多个内部状态。

    执行Generator函数会返回一个遍历器对象,也就是说,Generator函数除了状态机,还是一个遍历器对象生成函数。返回的遍历器对象,可以依次遍历Generator函数内部的每一个状态。

    形式上

    • Generator函数是一个普通函数,但是有两个特征。
      • function关键字与函数名之间有一个星号*;
      • 函数体内部使用yield语句,定义不同的内部状态(yield语句在英语里的意思就是“产出”)。
    let tell=function* () {
        yield 'a';
        yield 'b';
        yield 'c';
    };
    let k=tell();
    console.log(k.next());
    console.log(k.next());
    console.log(k.next());
    console.log(k.next());
    

    上面代码定义了一个Generator函数tell,它内部有三个yield语句'a','b'和'c' ; 即该函数有三个状态:'a','b'和'c'语句(结束执行)。

    总结一下,调用Generator函数,返回一个遍历器对象,代表Generator函数的内部指针。以后,每次调用遍历器对象的next方法,就会返回一个有着value和done两个属性的对象。value属性表示当前的内部状态的值,是yield语句后面那个表达式的值;done属性是一个布尔值,表示是否遍历结束。true:表示遍历结束,false:表示遍历没结束;

    yield

    需要注意的是,yield语句后面的表达式,只有当调用next方法、内部指针指向该语句时才会执行,因此等于为JavaScript提供了手动的“惰性求值”(Lazy Evaluation)的语法功能。

    function* gen() {
      yield  123 + 456;
    }
    

    上面代码中,yield后面的表达式123 + 456,不会立即求值,只会在next方法将指针移到这一句时,才会求值。

    yield语句注意事项

    • yield语句不能用在普通函数中,否则会报错。
    • yield语句如果用在一个表达式之中,必须放在圆括号里面。
    console.log('Hello' + yield); // SyntaxError
    console.log('Hello' + yield 123); // SyntaxError
    
    console.log('Hello' + (yield)); // OK
    console.log('Hello' + (yield 123)); // OK
    
    • yield语句用作函数参数或赋值表达式的右边,可以不加括号。
    foo(yield 'a', yield 'b'); // OK
    let input = yield; // OK
    

    暂缓执行函数

    Generator函数可以不用yield语句,这时就变成了一个单纯的暂缓执行函数。

    function* f() {
      console.log('执行了!')
    }
    
    var generator = f();
    
    setTimeout(function () {
      generator.next()
    }, 2000);
    

    上面代码中,函数f如果是普通函数,在为变量generator赋值时就会执行。但是,函数f是一个Generator函数,就变成只有调用next方法时,函数f才会执行。

    Generator 与 Iterator接口的关系

    任意一个对象的Symbol.iterator方法,等于该对象的遍历器生成函数,调用该函数会返回该对象的一个遍历器对象。

    由于Generator函数就是遍历器生成函数,因此可以把Generator赋值给对象的Symbol.iterator属性,从而使得该对象具有Iterator接口。

    //generator对象的新应用:给obj对象部署Iterator;
    let obj={};
    obj[Symbol.iterator]=function* () {
        yield 1;
        yield 2;
        yield 3;
    };
    for(let value of obj){
        console.log(value);
    }
    

    用Generator写抽奖

    let draw=function (count) {
        //具体抽奖逻辑
        console.info(`剩余${count}抽奖次数`);
    };
    let residue=function* (count) {
        while(count>0){
            count--;
            yield draw(count);
        }
    };
    let star=residue(5);
    let btn=document.createElement('button');
    btn.innerHTML='点击抽奖';
    document.body.appendChild(btn);
    btn.onclick=function () {
        star.next();
    }
    

    前端定时的去接收服务端的变化

    • 两种办法:1)websocket-兼容性不好 ; 2)常轮询-看如下代码
    let ajax=function* () {
        yield new Promise((resolve,reject)=>{
            setTimeout(function () {
                resolve({code:0})
            },200)
        })
    };
    let pull=function () {
        let generator=ajax();
        let step=generator.next();
        //拿回后台的数据
        step.value.then(function (d) {
            if(d.code!=0){
                setTimeout(function () {
                    console.info('wait');
                    pull();
                },1000)
            }else{
                console.log(d);
            }
        })
    };
    pull();
    

    相关文章

      网友评论

      本文标题:es6解读7- Generator

      本文链接:https://www.haomeiwen.com/subject/xlqckxtx.html