有没有办法测试闭包是否也是生成器?

2022-08-30 22:16:29

我正在使用一个PHP类,它需要接受多种类型的迭代器,并将它们包含在统一的包装器中。我需要支持(并且可以!)的迭代器类型之一是包含yield关键字的匿名函数 - 本质上是匿名生成器。

在PHP中,有没有办法测试匿名函数是否是生成器?以下是我尝试过的方法列表(旨在显示输出,而不是我如何使用它们):

$anon = function() { yield 1; yield 2; };   // build anonymous generator
gettype($anon);                             // "object"
$anon instanceof \Traversable;              // 0
$anon instanceof \Iterable;                 // 0
$anon instanceof \IteratorAggregate;        // 0
$anon instanceof \Generator;                // 0
$anon instanceof \Closure;                  // 1 -- but doesn't tell me whether or not the closure is actually generator

$anon = $anon();                            // invoke, then run the same checks
gettype($anon);                             // "object"
$anon instanceof \Traversable;              // 1 (!!!)
$anon instanceof \Iterable;                 // 0
$anon instanceof \IteratorAggregate;        // 0
$anon instanceof \Generator;                // 1 (!!!)
$anon instanceof \Closure;                  // 0

正如您在上面所看到的,我可以调用匿名函数,然后确定该函数是否为可遍历类型,但是为了以延迟加载的方式实现它(例如,围绕SQL语句调用的匿名函数包装器,后跟每条记录的收益),我无法在foreach迭代之前调用匿名函数。

PHP中是否有任何我缺少的方法/类型可用于确定尚未调用的匿名方法是否是生成器?


答案 1
$anon = function() { echo 'INVOKED', PHP_EOL; yield 1; yield 2; };
$r = new ReflectionFunction($anon);
var_dump($r->isGenerator());

显示

bool(true);

INVOKED根本不显示,证明在任何时候都不会调用闭包


答案 2

对于那些想知道的人,该匿名函数不是生成器。由于调用,它将返回生成器。

根据文档

首次调用生成器函数时,将返回内部生成器类的对象。


推荐