Я не думаю, что есть способ сделать это на одном графике с макетом graphviz twopi . Twopi должен в целом хорошо выполнять настройку root узлов каждого подграфа, поскольку, как указано в документах , он случайным образом выберет один из узлов, наиболее удаленных от конечный узел как root, поэтому в случае наличия одного root узла это должно привести к ожидаемому топологическому расположению. Хотя, если это не так, и вы хотите установить вручную корни для каждого подграфа, я бы подошел к этому, перебирая графы, связанные с подграфами компонентов, и отображая каждый компонент в отдельности. ось на графике подзаговора, создавая для каждого из них пользовательский graphviz_layout
.
Вот как это можно сделать, используя следующий пример графика:
from matplotlib import pyplot as plt
import pygraphviz
from networkx.drawing.nx_agraph import graphviz_layout
result_set = {('plant','tree'), ('tree','oak'), ('flower', 'rose'), ('flower','daisy'), ('plant','flower'), ('tree','pine'), ('plant','roots'), ('animal','fish'),('animal','bird'), ('bird','robin'), ('bird','falcon'), ('animal', 'homo'),('homo','homo-sapiens'), ('animal','reptile'), ('reptile','snake'),('fungi','mushroom'), ('fungi','mold'), ('fungi','toadstool'),('reptile','crocodile'), ('mushroom','Portabello'), ('mushroom','Shiitake'),('pine','roig'),('pine','pinyer'), ('tree','eucaliptus'),('rose','Floribunda'),('rose','grandiflora')}
G=nx.from_edgelist(result_set, create_using=nx.DiGraph)
чтобы перебрать существующие подграфы, мы должны создать копию текущего графа в виде неориентированного графа, если он еще не создан, и создать список подграфов, используя nx.connected_component_subgraphs
:
UG = G.to_undirected()
subgraphs = list(nx.connected_component_subgraphs(UG))
Допустим, мы знаем, что мы хотим, чтобы узлы root различных компонентов были узлами 'plant'
, 'animal'
и 'mushroom'
, теперь мы можем создать набор вспомогательных участков и выполнять итерации по соответствующим осям. вместе с объектами подграфа и списком корней (следя за тем, чтобы они были в одинаковом порядке), создавая новый макет для каждого подграфа, устанавливая соответствующие root узлы:
n_cols = 2
roots = ['plant','animal','mushroom']
fig, axes = plt.subplots(nrows=int(np.ceil(len(subgraphs)/n_cols)),
ncols=n_cols,
figsize=(15,10))
plt.box(False)
for subgraph, root, ax in zip(subgraphs, roots, axes.flatten()):
pos = graphviz_layout(G, prog='twopi', args=f"-Groot={root}")
nx.draw(subgraph, pos=pos, with_labels=True,
node_color='lightblue', node_size=500, ax=ax)