PHP RecursiveIteratorIterator和嵌套集

时间:2009-02-05 21:23:32

标签: php recursion iterator nested-sets

我在一个层次结构中有一组对象。有一个顶级“根”节点,它有子节点,后面又有子节点等。我试图使用嵌套集模型将这个结构保存到数据库中,每个节点的每个“边”都编号为定义层次结构,如Managing Hierarchical Data in MySQL

alt text
(来源:mysql.com

我的问题是计算左右值。我通常使用RecursiveIteratorIterator迭代层次结构,但我无法计算如何计算数字而不诉诸通过引用解析索引变量的递归函数。

有什么想法吗?

这可能没用,但这是我目前的(错误的)代码:

$iterator = new RecursiveIteratorIterator(
    new Node_List(array($root)),
    RecursiveIteratorIterator::SELF_FIRST);

$i = 0;     
foreach ($iterator as $node) {
    $node->left = ++$i;
    $node->right = ++$i;
}

正如你所看到的那样,会产生这样的结果:

Node 
    Node 
    Node 

左右值:

Node (1, 2)
    Node (3, 4)
    Node (5, 6)

什么时候应该:

Node (1, 6)
    Node (2, 3)
    Node (4, 5)

2 个答案:

答案 0 :(得分:4)

我明白了,这是解决方案(简化):

$iterator = new RecursiveIteratorIterator(
    new Site_Node_List(array($root)),
    RecursiveIteratorIterator::SELF_FIRST);

$sides = array();
$s = 0;
$i = 0;
$parents = array();
foreach ($iterator as $item) {
    $js = array_splice($parents, $depth, count($parents), array($i));
    foreach (array_reverse($js) as $j) {
        $sides[$j]['right'] = ++$s;
    }
    $sides[$i]['left'] = ++$s;
    $i++;
}
foreach (array_reverse($parents) as $j) {
    $sides[$j]['right'] = ++$s;
}

这是我的实际代码的简化版本,因为它只是将“side”值存储在一个单独的数组中,但它证明了原理。

基本思想是将所有父节点(由深度值跟踪)存储在一个数组中,并且只在循环中写入“left”值。然后,当深度减小时,意味着您已经返回层次结构,因此将父对象拼接以移除不再相关的对象,并将它们循环(反向)设置“正确”值。最后,你必须在最后循环其余的父母。

答案 1 :(得分:0)

没有递归就无法解决这个问题。您需要以下内容:

function tag_recursive($node, &$number) {
    $node->left = $number++;
    foreach ($node->children as &$child) {
        tag_recursive($child, $number);
    }
    $node->right = $number++;
}

function tag($node) {
    $number = 1;
    tag_recursive($node, $number);
    // $number is now highest id + 1
}