Django-mptt完全越野车或我做错了吗?

时间:2010-01-28 07:12:54

标签: django mptt django-mptt

我试图用django-mptt运气很少。这是与Python2.5,windows,sqlite3,Django 1.2pre,django-mptt最新的svn。

代码:

模型:

class Node(models.Model):
    name   = models.CharField(max_length=20, blank=True)
    parent = models.ForeignKey('self', null=True, blank=True, related_name='children')

    def __unicode__(self):
        return self.name

mptt.register(Node)

设置:

nodes = []
for i in range(15):
    n = Node(name='node'+str(i))
    n.save()
    nodes.append(n)

nodes[0].move_to(None)
nodes[0].save()
for n in range(1,15):
    nodes[n].move_to(nodes[(n-1)/2],'last-child')
    nodes[n].save()

这应该创建一个树,其中一个根和两个子节点悬挂在每个非叶节点上。

现在好玩的开始了:

>>> nodes[0].children.all()
[<Node: node1>, <Node: node2>]
>>> nodes[0].get_descendants()
[]

>>> nodes[0].get_descendants(True)
[<Node: node0>, <Node: node2>]


>>> for n in nodes:
...     print n, n.get_ancestors()
...
node0 []
node1 [<Node: node0>]
node2 [<Node: node0>]
node3 [<Node: node0>, <Node: node2>]
node4 [<Node: node0>, <Node: node2>]
node5 [<Node: node0>, <Node: node2>]
node6 [<Node: node0>, <Node: node2>]
node7 [<Node: node0>, <Node: node2>, <Node: node6>]
node8 [<Node: node0>, <Node: node2>, <Node: node6>]
node9 [<Node: node0>, <Node: node2>, <Node: node6>]
node10 [<Node: node0>, <Node: node2>, <Node: node6>]
node11 [<Node: node0>, <Node: node2>, <Node: node6>]
node12 [<Node: node0>, <Node: node2>, <Node: node6>]
node13 [<Node: node0>, <Node: node2>, <Node: node6>]
node14 [<Node: node0>, <Node: node2>, <Node: node6>]

为什么这么多祖先错了?例如,节点10应该有祖先,(0,1,10)

我做错了什么或者django-mptt中有错误吗?

1 个答案:

答案 0 :(得分:14)

我不会说这是有缺陷的,但是你需要注意一些问题。

将子项添加到父项时,子项的树属性会使用MPTT特定的lftrghtlevel值正确更新。

但是,django-mptt 更新您持有的父版本。数据库中的版本已更新,但本地变量中的副本不是(请记住Django模型实例没有标识,因此当数据库或引用同一数据库行的其他实例更新时不要更新)。

这意味着您添加到父对象的下一个子项将获得错误的左右值,如果您随后保存父级也将具有错误的值。

解决方案是每次添加子项时从数据库重新加载父项:

for n in range(1,15):
    parent_pos = (n-1)/2
    parent = nodes[parent_pos]
    nodes[n].move_to(parent, 'last-child')
    nodes[n].save()
    nodes[parent_pos] = Node.objects.get(pk=parent.pk)