zoukankan      html  css  js  c++  java
  • Generator函数(二)

    for...of循环

    1.for...of循环可以自动遍历Generator函数,不需要再调用next方法

    function* helloWorldGenerator(){
        yield 'hello';
        yield 'world';
        return 'ending';
    }
    for (let f of helloWorldGenerator()){
        console.log(f);//'hello','world'
    }
    
    var hw = helloWorldGenerator();
    hw.next();
    hw.next();//done的值是false
    hw.next();//done的值是true,所以此时不会再输出ending
    

    这里使用for...of循环和next()方法遍历Generator函数,在使用for...of循环可以自动遍历Generator函数,且此时不再需要调用next方法。
    而且遍历出来的结果也不同。
    2.前面章节介绍过,for...of循环,扩展运算符(...),解构赋值和array.from方法内部调用的都是遍历器接口。这就是说,他们可以将Generator函数返回的Iterator对象作为参数。

    function* numbers(){
        yield 1;
        yield 2;
        return 3;
        yield 4;
    }
    
    [...numbers()];//[1,2],扩展运算符(...)
    Array.from(numbers()) //[1,2],Array.from方法
    let [x,y] = numbers(); //解构赋值
    x//1
    y//2
    
    for(let n of numbers()){
        console.log(n);
    }
    //1
    //2
    

    3.使用Generator函数和for...of循环为原生的JavaScript对象加上遍历接口

    function* objectEntries(obj){
        let propKeys = Reflect.ownKeys(obj);//获取一个对象obj的key
        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
    

    第二种方法实现添加遍历接口

        function* objectEntries(obj){
            let propKeys = Reflect.ownKeys(obj);
            for(let propkey of propKeys){
                yield [propkey,obj[propkey]];
            }
        }
        let jane = {first:'jane',last:'doe'};
        jane[System.iterator] = objectEntries;
        for(let [key,value] of jane){
            console.log(`${key}:${value}`);
        }
        //  first:jane
        //  last:doe
    

    Generator.prototype.throw()

    1.Generator函数返回的遍历器都有一个throw方法,可以在函数体外抛出错误,在函数体内捕获错误。

    var g = function* (){
        while (true ){
            try{
                yield;
            }catch (e){
                if(e != 'a') throw e;
                console.log('内部捕获',e);
            }
        }
    };
    var i = g();
    i.next();
    
    try{
        i.throw('a');
        i.throw('b');
    }catch(e){
        console.log('外部捕获',e);
    }
    //内部捕获 a
    //外部捕获 b
    

    总结一下,有几种错误抛出和捕获的情况。
    (1)Generator函数体内有try...catch块,函数体外也有try...catch块,如果函数体外用多个throw方法抛出错误:
    解决错误的情况:第一个throw方法,将被函数体内的捕获,后面的throw方法会被函数体外的捕获。(因为Generator函数内部的catch语句已经执行过了,不会再捕获到后面的throw方法抛出的错误)
    (2)Generator函数体内和体外都有try...catch块,如果函数体外用throw命令抛出错误:
    解决情况:因为用throw命令抛出的错误,只能被函数体外的catch捕获,即使函数体内有catch块。
    (3)Generator函数体内没有try...catch块,函数体外有try...catch块,如果函数体外用throw方法或者throw命令抛出错误:
    解决情况:被外部的try...catch块捕获。
    (4)Generator函数体内和体外都没有try...catch块,那么如果抛出错误:程序会报错,直接中断执行。

    2.关于Generator函数体内有无try...catch块,会不会影响到遍历器的状态,即关系到next方法会不会被正常执行。
    如果Generator函数体内没有try...catch块,则遍历器的throw方法抛出的错误不会影响到下一次的遍历,否则遍历直接终止,但是如果用的是throw命令,则不会影响到遍历器的状态。

    var gen = function* gen(){
        yield console.log('hello');
        yield console.log('world');
    }
    
    var g = gen();
    g.next();
    try {
        g.throw();
    }catch(e){
        g.next();
    }
    //hello
    

    虽然Generator函数内部没有部署try...catch块,那么throw方法抛出的错误即使会被外部的try...catch代码块捕获。
    但是第二次调用next()方法的时候遍历器状态已经变成终止了。但是使用throw命令(throw new Error())抛出的错误就不会影响到遍历器的状态,即next()方法。
    3.Generator函数这种函数体内捕获错误的机制大大方便了对错误的处理。如果使用回调函数,想要捕获多个错误的话,就不得不为每个函数写一个错误处理语句。
    但是使用Generator函数就会简化很多,例如

        function* g(){
            try{
                var a = yield foo('a');
                var b = yield foo('b');
                var c = yield foo('c');
            }catch (e){
                 console.log(e);
            }
            console.log(a,b,c);
        }
    

    4.Generator函数体内抛出的错误也可以被函数体外的catch捕获。

        function* foo(){
            var x = yield 3;
            var y = x.toUpperCase();//这里报错
            yield y;
        }
        var it = foo();
         it.next();
    
        try{
            it.next(42);
        }catch(err){
            console.log(err);//这里捕获
        }
    
  • 相关阅读:
    绘图QPainter-画刷
    绘图QPainter-画笔
    pyqt5-多线程QThread类
    升级时出现错误的解决办法
    打包pyinstaller
    多文档界面QMdiArea
    停靠窗口QDockWidget
    堆叠窗口QStackedWidget
    VC运行库版本不同导致链接.LIB静态库时发生重复定义问题的一个案例分析和总结
    【一】ODB
  • 原文地址:https://www.cnblogs.com/sminocence/p/7274984.html
Copyright © 2011-2022 走看看