7

我正在["abc", "abd", "aec", "add", "adcf"]使用 python3 的 anytree 包从列表中创建一棵树。在这棵树中,每个列表元素的第一个字符 -a是一个根,随后,其他字符被添加为它们的子元素。当我渲染树时,它看起来像:

a
├── b
│   ├── c
│   └── d
├── e
│   └── c
└── d
    ├── d
    └── c
        └── f

但是当我使用方法将树渲染为图片时to_picture,图像是 -

在此处输入图像描述

我不希望合并公共节点,因为它正在向我的树添加不需要的路径。

4

2 回答 2

10

graphviz节点使用它们的 id排列。在您的情况下,该图仅使用节点名称生成,然后graphviz在您获得它时创建循环边缘。

您真正想要的是id每个节点和label与之关联的节点都不同。该类DotExporter有一个名为的属性nodeattrfunc,我们可以向其传递函数或 lambda 并为节点生成属性

以下是根据您的数组执行此操作的方法

import anytree
from anytree import Node, RenderTree

data = ["abc", "abd", "aec", "add", "adcf"]
from anytree.exporter import DotExporter

nodes = {}
first_node = None

for elem in data:
    parent_node = None
    parent_node_name = ""
    for i, val in enumerate(elem):
        if i not in nodes:
            nodes[i] = {}
        key = parent_node_name + val
        if key not in nodes[i]:
            print("Creating new node for ", key)
            nodes[i][key] = Node(key, parent=parent_node, display_name=val)

        if first_node is None:
            first_node = nodes[i][key]
        parent_node = nodes[i][key]
        parent_node_name = val

print(nodes)
DotExporter(nodes[0]["a"],
            nodeattrfunc=lambda node: 'label="{}"'.format(node.display_name)).to_dotfile("graph.txt")
DotExporter(nodes[0]["a"],
            nodeattrfunc=lambda node: 'label="{}"'.format(node.display_name)).to_picture("graph.png")

这将生成以下点文件

digraph tree {
    "a" [label="a"];
    "ab" [label="b"];
    "bc" [label="c"];
    "bd" [label="d"];
    "ae" [label="e"];
    "ec" [label="c"];
    "ad" [label="d"];
    "dd" [label="d"];
    "dc" [label="c"];
    "cf" [label="f"];
    "a" -> "ab";
    "a" -> "ae";
    "a" -> "ad";
    "ab" -> "bc";
    "ab" -> "bd";
    "ae" -> "ec";
    "ad" -> "dd";
    "ad" -> "dc";
    "dc" -> "cf";
}

和下图

最终图表

于 2018-03-23T05:33:34.057 回答
1

作为anytree to_picture()函数使用 graphviz 生成文件。

这里=类似的问题,您可以阅读:

graphviz 使用节点 ID 作为标签。如果不同的节点需要具有相同的标签,则必须明确定义标签。

于 2018-03-22T16:42:56.513 回答