Where Argument Of First Next() Call Goes?
Solution 1:
The next
method is defined as follows:
25.3.1.2 Generator.prototype.next ( value )
The
next
method performs the following steps:
- Let g be the this value.
- Return GeneratorResume(g, value).
The GeneratorResume abstract operation uses value at step 10:
25.3.3.3 GeneratorResume ( generator, value )
The abstract operation GeneratorResume with arguments generator and value performs the following steps:
- Let genContext be the value of generator’s [[GeneratorContext]] internal slot.
- Resume the suspended evaluation of genContext using NormalCompletion(value) as the result of the operation that suspended it. Let result be the value returned by the resumed computation.
The first possibility is that the evaluations was suspended by the use of yield
(i.e. "suspendedYield"
state).
Its behavior is explained in 14.4.14 Runtime Semantics: Evaluation:
YieldExpression :
yield
- Return GeneratorYield(CreateIterResultObject(undefined, false)).
(Analogous for YieldExpression : yield
AssignmentExpression)
The GeneratorYield abstract operation suspends the generator as follows:
Set the code evaluation state of genContext such that when evaluation is resumed with a Completion resumptionValue the following steps will be performed:
- Return resumptionValue.
- NOTE: This returns to the evaluation of the YieldExpression production that originally called this abstract operation.
So the value passed as the argument of the second next
is used as the returned value of the first yield
expression. And the value passed as the argument of the 3rd next
is used as the returned value of the 2nd yield
expression. And so on.
However, there is also the possibility that the generator hadn't started yet (i.e. "suspendedStart"
state).
This is done by the GeneratorStart abstract operation:
- Set the code evaluation state of genContext such that when evaluation is resumed for that execution context the following steps will be performed:
But those "following steps" don't use the resumption value.
Therefore, the value passed as the argument of the first next
is discarded.
Solution 2:
Nope, you can't use that first value. It's instructive if you change your code to :
'use strict';
function *generate(arg) {
console.log(arg)
for(let i = 0; i < 3;i++) {
console.log(yield i);
}
return 'done';
}
var gen = generate('arg');
console.log(gen.next('a'));
console.log(gen.next('b'));
console.log(gen.next('c'));
console.log(gen.next('d'));
When you instantiate the generator, it does not start executing yet and nothing is logged. On the first gen.next('a')
, you run up to the first yield which goes through the console.log(arg)
in the generator and then does yield 0
. This then gets console.log
in the caller and you get {value: 0, done:false}
, etc. until you complete the iteration. The overall output looks as follows:
arg
{ value: 0, done: false }
b
{ value: 1, done: false }
c
{ value: 2, done: false }
d
{ value: 'done', done: true }
The final done
is the return value and will be undefined
if you omit the return
on the generator.
Post a Comment for "Where Argument Of First Next() Call Goes?"