Is it possible to reset an ECMAScript 6 generator to its initial state?

前端 未结 8 1341
有刺的猬
有刺的猬 2020-12-01 12:19

Given the provided (very simple) generator, is it possible to return the generator back to its original state to use again?

var generator = function*() {
            


        
相关标签:
8条回答
  • 2020-12-01 12:23

    Whenever you need to "reset" an iterable, just toss the old one away and make a new one.

    var generator = function*() {
        yield 1;
        yield 2;
        yield 3;
    };
    const makeIterable = () => generator()
    
    for (let x of makeIterable()) {
        console.log(x);
    }
    
    // At this point, iterable is consumed.
    // Is there a method for moving iterable back
    // to the start point by only without re-calling generator(),
    // (or possibly by re-calling generator(), only by using prototype 
    //  or constructor methods available within the iterable object)
    // so the following code would work again?
    
    for (let x of makeIterable()) {
        console.log(x);
    }

    0 讨论(0)
  • 2020-12-01 12:32

    At this point, iterable is consumed.

    Which means its internal [[GeneratorState]] is completed.

    Is there a method for moving iterable back to the start point by only without re-calling generator()

    No. The spec states

    Once a generator enters the "completed" state it never leaves it and its associated execution context is never resumed. Any execution state associated with generator can be discard at this point.

    or possibly by re-calling generator(), only by using prototype or constructor methods available within the iterable object

    No. While not explicitly stated in the spec, there are no more instance-specific properties available on the iterable object than [[GeneratorState]] and [[GeneratorContext]].

    However, the informative "Generator Object Relationships" grapic states:

    Each Generator Function has an associated prototype that does not have a constructor property. Hence a generator instance does not expose access to its generator function.

    I would like to be able to pass the iterable off to some other scope

    Pass the generator function instead. Or something that yields new generator instances.

    0 讨论(0)
  • 2020-12-01 12:36

    As best I can tell that isn't possible. Per this useful wiki and the draft version of ES6 on generators, once you've returned from it (rather than yielded), it puts it into the "closed" state and there is no way to move it back to the "newborn" state which is how a new generator starts out.

    You may have to pass along a callback to your other scope for creating a new generator. As a work-around, you could even add that callback as a custom method on the generator you sent to the other scope if you wanted and that callback would create a new generator for the other scope.

    If you think about how generators work, they'd have to execute over from scratch to reset their initial state and there's simply no reason to support that. That would be analagous to asking why you can't just re-execute the constructor on an existing object and expect to have a virgin object in the same object. While it's all technically doable, it's hairy to make work right and there's really no reason to support it. If you want a virgin object, just create a new one. Same with a generator.


    This is a bit of a hack, but a curious thing to contemplate. You could make a generator that repeated itself. Suppose your generator worked like this:

    var generator = function*() {
        while (true) {
            yield 1;
            yield 2;
            yield 3;
            yield null;
        }
    };
    
    var iterable = generator();
    
    for (let x of iterable) {
        if (x === null) break;
        console.log(x);
    }
    
    // generator is now in a state ready to repeat again
    

    I can easily see how this might be an anti-pattern though because if you ever do this:

    for (let x of iterable) {
        console.log(x);
    }
    

    You will have an infinite loop, so it would have to be used with great care. FYI, the above wiki shows examples of an infinite Fibonacci sequence so an infinite generator is certainly contemplated.

    0 讨论(0)
  • 2020-12-01 12:39

    If your intention is

    to some other scope, iterate over it, do some other stuff, then be able to iterate over it again later on in that same scope.

    Then the only thing you shouldn't try doing is passing the iterator, instead pass the generator:

    var generator = function*() {
        yield 1;
        yield 2;
        yield 3;
    };
    
    var user = function(generator){
    
        for (let x of generator()) {
            console.log(x);
        }
    
        for (let x of generator()) {
            console.log(x);
        }
    }
    

    Or just make a "round robin" iterator and check while iterating

    var generator = function*() {
        while(true){
            yield 1;
            yield 2;
            yield 3;
        }
    };
    
    for( x in i ){
        console.log(x);
        if(x === 3){
            break;
        }
    }
    
    0 讨论(0)
  • 2020-12-01 12:39

    No there is no going back to same state.

    To make it clear, you must understand the working of generator functions.

    When generator function gets called first time, it returns iterator(as its entire body). Initial status of this return iterator is get stored in its variables. Two very important variables are GeneratorStatus, GeneratorLocation.

    There are are other variables such as GeneratorFunction, GeneratorReceiver, Scopes. Which can ignored to understand this answer.

    So initial status will be, GeneratorStatus : suspended. GeneratorLocation: 1;

    Now to use iterator you should call it using .next(); Hence forth iterator will resume its execution from location pointed by 'GeneratorLocation'

    Now generator will update its value of GeneratorLocation to line no where it first yields the result and GeneratorLocation will be same until it returns last yield.

    Now for Each successive calls of .next, Generator will resume its execution from value of GeneratorLocation and not from the beginning.

    Hence, Unless you repeat the code in generator function, resetting to initial state is not possible. best solution is just recreate new iterator with params.

    0 讨论(0)
  • 2020-12-01 12:43

    You can also have your generator reset your iterable like this:

    let iterable = generator();
    
    function* generator(){
        yield 1;
        yield 2;
        yield 3;
        iterable = generator();
    }
    
    for (let x of iterable) {
        console.log(x);
    }
    
    //Now the generator has reset the iterable and the iterable is ready to go again.
    
    for (let x of iterable) {
        console.log(x);
    }
    

    I do not personally know the pros and cons of doing this. Just that it works as you would expect by reassigning the iterable each time the generator finishes.

    EDIT: With more knowledge of how this work I would recommend just using the generator like Azder Showed:

    const generator = function*(){
        yield 1;
        yield 2;
        yield 3;
    }
    
    for (let x of generator()) {
        console.log(x);
    }
    
    for (let x of generator()) {
        console.log(x);
    }
    

    The version I recommended will prevent you from being able to run through the iterations if it ever fails... For example if you were waiting on one url to call another. If the first url fails you would have to refresh your app to before it would be able to try that first yield again.

    0 讨论(0)
提交回复
热议问题