0

我在尝试向上或向下移动子节点或父节点时遇到了麻烦……数学不太好。

CREATE TABLE IF NOT EXISTS `pages` (   `page-id` mediumint(8) unsigned

NOT NULL AUTO_INCREMENT, page-left mediumint(8) unsigned NOT NULL,
page-rightsmallint(8) unsigned NOT NULL, page-titletext NOT NULL,
page-contenttext NOT NULL,
page-timeint(11) unsigned NOT NULL, page-slugtext NOT NULL,
page-templatetext NOT NULL,
page-parentmediumint(8) unsigned NOT NULL, page-type文本 NOT NULL, PRIMARY KEY ( page-id) ) ENGINE=MyISAM DEFAULT CHARSET=latin1 ;

插入pages( page-id, page-left, page-right, page-title, page-content, page-time, page-slug, page-template, page-parent, page-type) 值 (17, 1, 6, '1', '', 0, 'PARENT', '', 0, ''), (18, 2, 5, '2', '', 0, 'SUB', '', 17, ''), (19, 3, 4, '3', '', 0, 'SUB-SUB', '', 18, ' '), (20, 7, 8, '5', '', 0, 'TEST', '', 0, '');

例如,我将如何将 TEST 向上移动到 PARENT 上方,并通过使用 page-left/page-right ID 将 SUB 向下移动到 SUB-SUB 下方?不需要代码,只需要帮助它的 SQL 概念或数学,它可以帮助我理解如何更好地移动它......

4

1 回答 1

3

所以基本上你想将邻接列表转换为嵌套集?首先更新您的邻接列表(即,将 page_parent 值更新为新树的正确值),然后运行下面的转换。

使用 PHP(基本代码,未经测试):

class Tree
{    
    private $count = 0;
    private $data = array();

    /**
     * Rebuild nested set
     * 
     * @param $rawData array Raw tree data
     */
    public function rebuild($rawData)
    {
        $this->data = $rawData;
        $this->count = 1;
        $this->traverse(0);        
    }

    private function traverse($id)
    {
        $lft = $this->count;
        $this->count++;

        if (isset($this->data[$id])) {
            $kid = $this->data[$id];
            if ($kid) {
                foreach ($kid as $c) {
                    $this->traverse($c);
                }
            }
        }

        $rgt = $this->count;
        $this->count++;

        // TODO: Update left and right values to $lft & $rgt in your DB for page_id $id
        ...
    }
}

当你调用它时,$rawData 应该包含一个由 parent-id 索引的 ID 数组,你可以按如下方式创建它(基于你的表结构)($db 应该包含一个活动的 PDO 连接对象):

    $sql = 'SELECT page_id, page_parent FROM pages ORDER BY page_parent';

    $stmt = $db->prepare($sql);
    $rawData = array();
    $stmt->execute();
    while ($row = $stmt->fetch()) {
        $parent = $row['page_parent'];
        $child = $row['page_id'];
        if (!array_key_exists($parent, $rawData)) {
            $rawData[$parent] = array();
        }
        $rawData[$parent][] = $child;
    }

要进行转换,您需要以下内容:

$tree = new Tree();
$tree->rebuild($rawData);

因此,基本上您创建一个数组,其中包含由父级索引的树中的所有节点,该数组将以递归方式遍历以确定每个节点的正确左值和右值。

BTW You could do it in plain SQL (after you adapt table/column names) : http://bytes.com/topic/mysql/answers/638123-regenerate-nested-set-using-parent_id-structure

于 2010-03-12T13:31:42.577 回答