2

我有一个需要迭代的抽象语法树。AST 由柠檬移植到 PHP生成。

现在“通常”,我会使用全新的闪亮 (PHP 5.3.1) SPL 类,它看起来像这样:

$it = new \RecursiveIteratorIterator(
  new \RecursiveArrayIterator($ast['rule']),
  \RecursiveIteratorIterator::SELF_FIRST);

实际上,这就是我已经在代码的另一部分中所做的,它确定了整个树的粗略类型(即它可以是赋值、条件等)。现在抛开细节不谈,唯一重要的是迭代是做RecursiveIteratorIterator::SELF_FIRST,也就是自顶向下。

回到我的问题,我需要自下而上地迭代 AST,即类似 RecursiveIteratorIterator::CHILD_FIRST 的东西,以便在树中进行一些替换和优化。

问题是,这些操作需要是上下文感知的,即我需要到当前节点的路径。而且由于我想自下而上迭代,所以我不能使用 RecursiveIteratorIterator。

好好想一想。我想自下而上迭代并在每次迭代时拥有当前节点的自上而下上下文(堆栈)。从技术上讲,它应该是可能的,因为 RecursiveIteratorIterator 必须首先到达树的尾部,才能向后迭代。在到达尾部的过程中,它可以缓存当前位置,并在从递归返回时简单地弹出元素。

现在这是一个关键字:缓存。这就是为什么我怀疑另一个 SPL 类应该是可能的:RecursiveCachingIterator。

问题是:真的有可能吗?如果是,如何?

我一直试图用一些代码来解决问题,但没有成功,而且文档很少。真的,真的很稀缺。

谁使用 SPL 找到了最优雅的解决方案,脱帽致敬!你是 PH​​P 大师!

PS:如果不清楚,我正在寻找尽可能多的 SPL(重新)使用。我知道我可以使用自定义堆栈编写自己的递归函数,无需提醒我。

4

1 回答 1

2

我已经设法通过继承 RecursiveIteratorIterator 并分别在 ::endChildren() 和 ::callGetChildren 中管理堆栈来使其工作。也许这会对某人有所帮助。向我自己致敬:-)

于 2010-02-03T15:25:25.367 回答