php - 在 PHP 中:“return”、“yield”、“yield from”和在同一个函数中混合 yield 和 return 有什么区别?
问题描述
return
和之间的区别yield
似乎很清楚,直到我发现也有yield from
可能将两者结合起来return
和yield
在同一个功能中!
我的理解return
是后面的一切都没有执行,对吧?
然而:
function generate(): iterable {
return [1, 2, 3];
}
foreach (generate() as $value) {
echo $value;
}
产生:“123”
但以下内容:
function generate(): iterable {
return [1, 2, 3];
yield;
}
foreach (generate() as $value) {
echo $value;
}
什么都不生产!那么这意味着yield被执行了吗?
这是一个错误吗?
解决方案
Return
只需将唯一值返回给调用者。
Yield
将当前函数/方法转换为返回 a Generator
,这将产生多个唯一值:每次yield
触发时,它都会将值提供给调用者,一次一个,传统上使用foreach
循环。
Yield
+Return
生成器除了生成值之外,还可以提供唯一的返回值。该值不会成为生成器循环的一部分,必须使用Generator::getReturn()
方法访问它。
Return
+Yield
这可能被视为一个错误,但事实并非如此。
它们是两个阶段:
- 从代码到字节码:在这个阶段,
generate()
函数被视为包含yield
关键字,因此它被标记为产生一个Generator
. - 执行:因为
return
恰好在 之前yield
,所以生成器没有机会产生任何值。但是,[1, 2, 3]
可以使用 检索该数组Generator::getReturn()
。
一个完整的注释示例:
// Generate integers 1 and 2
function generateIntegers1And2(): Generator {
yield 1; // <--+ <--+ <--+
yield 2; // <-+ <-+ <-+
} // | | |
// | | |
foreach (generateIntegers1And2() as $value) { // | | |
var_dump($value); // Shows 1, then 2 ->* | |
} // | |
// | |
function generateOuterYield(): Generator { // | |
// Yields the generator *itself* returned by | |
// generateIntegers1And2() not the actual values | |
// generated by it. | |
// This means we are producing here a generator | |
// of generator of integers. | |
yield generateIntegers1And2(); // <-+ | |
} // | | |
// | | |
foreach (generateOuterYield() as $value) { // | | |
var_dump($value); // ->* | |
// The two levels of imbrication means we have | |
// to loop once more to actually consume | |
// generateIntegers1And2 | |
foreach ($value as $val) { // | |
var_dump($val); // Shows 1, then 2 ->* |
} // |
} // |
// |
// A generator can just be returned as-is: |
function generateOuterReturn(): Generator { // |
return generateIntegers1And2(); // |
} // |
// |
// it doesn't change the way it is consumed |
foreach (generateOuterReturn() as $value) { // |
var_dump($value); // Shows 1, then 2 |
} // |
// |
function generateOuterYieldFrom(): Generator { // |
// First yield values generated by generateIntegers1And2() |
yield from generateIntegers1And2(); // *<---+
// then yield integers 3 |
yield 3; // <--+
// and 4 |
yield 4; // <-+
} // |
// |
foreach (generateOuterYieldFrom() as $value) { // |
var_dump($value); // Shows 1, 2, 3 and 4 ->*
}
function generateIntegers56AndReturn(): Generator {
yield 5; // <---+
yield 6; // <--+
// |
return ["five", "six"]; // <--+ |
} // | |
// | |
$gen = generateIntegers56AndReturn(); // | |
// | |
// Consume the values **yielded** by | |
// generateIntegers56AndReturn() | |
foreach ($gen as $value) { // | |
var_dump($value); // Shows 5, then 6 | ->*
} // |
// |
// Access the value **returned** by the generator |
var_dump($gen->getReturn()); // ->*
function wtf(): Generator {
return ["W", "T", "F", "!"];
// Without the following line, PHP would complain with a TypeError:
// Return value of wtf() must be an instance of Generator, array returned.
// The presence of a yield keyword anywhere inside the function makes it a Generator.
// However, since we return *before* reaching any *yield*, 42 is never yielded.
// This is empty generator!
yield 42;
}
$gen = wtf();
// This foreach loop is not entered!
foreach ($gen as $value) {
var_dump($value);
}
// However, we can loop on the array *returned* by wtf():
foreach ($gen->getReturn() as $value) {
echo $value; // Will print: WTF!
}
推荐阅读
- radio-button - NSIS RadioButton !insertmacro RadioGetChecked
- python - 我可以在旧版本的 tensorflow 上安装时尚 mnist 吗?
- mysql - 想从表 mysql laravel 中获取准确的记录
- r - 如何获得 R 中解释的主成分百分比方差?prcomp() 和 preProcess() 比较
- javascript - 如何将 React 和 material-ui 与烧瓶后端一起使用?
- regex - Hive 从字符串中提取数值
- jquery - 2位或3位电话号码区号的多个掩码,其余为7位
- google-sheets - 谷歌查询公式从下拉列表中包含“全部”
- javascript - 检测是否通过左键或右键单击选择选项
- youtube-api - 有没有办法通过 YouTube API 搜索任何视频中使用的音乐?