gpt4 book ai didi

python - NetworkX 的所有最低共同祖先

转载 作者:塔克拉玛干 更新时间:2023-11-03 03:57:06 35 4
gpt4 key购买 nike

我想从节点“a”和节点“o”获取所有 LCA 节点。

在这个有向图中,节点"l"和节点"m"是LCA节点。

下面是代码。

import networkx as nx

def calc_length(Graph, node1, node2, elem):
length1 = nx.shortest_path_length(Graph, node1, elem)
length2 = nx.shortest_path_length(Graph, node1, elem)
length_sum = length1 + length2
return length_sum

G = nx.DiGraph() #Directed graph
G.add_nodes_from(["a","b","c","d","e","f","g","h","i","j","k","l","m","n","o","p"])
edges = [("a","b"),("b","c"),("b","d"),("a","e"),("a","h"),("e","f"),("e","g"),("e","i"),("h","l"),("h","m"),("g","j"),("o","p"),("o","n"),("n","m"),("n","l"),("n","k"),("p","j"),]
G.add_edges_from([(e[0], e[1]) for e in edges])

preds_1 = nx.bfs_predecessors(G, "a")
preds_2 = nx.bfs_predecessors(G, "o")
common_preds = set([n for n in preds_1]).intersection(set([n for n in preds_2]))
common_preds = list(common_preds)
dic ={}
for elem in common_preds:
length_sum = calc_length(G, "a", "o", elem)
dic[elem] = length_sum

min_num = min(dic.values())
for k, v in sorted(dic.items(), key=lambda x:x[1]):
if v != min_num:
break
else:
print k, v

我想要更快的执行速度。

如果您有比前面提到的更好的方法来解决问题,请告诉我。

我会很感激你的帮助。

最佳答案

这里有几个问题,我已经在评论中指出了其中的一些问题。部分问题在于命名法令人困惑:最低共同祖先(as defined on wikipedia 并且大概在一般计算机科学中)真的应该命名为最低共同后代,以符合 networkx(以及任何理智的网络科学家)使用的命名法据我所知)。因此,您的广度优先搜索应该真正跟随后代,而不是前辈。下面实现这样的 LCA 搜索:

import numpy as np
import matplotlib.pyplot as plt; plt.ion()
import networkx as nx

def find_lowest_common_ancestor(graph, a, b):
"""
Find the lowest common ancestor in the directed, acyclic graph of node a and b.
The LCA is defined as on

@reference:
https://en.wikipedia.org/wiki/Lowest_common_ancestor

Notes:
------
This definition is the opposite of the term as it is used e.g. in biology!

Arguments:
----------
graph: networkx.DiGraph instance
directed, acyclic, graph

a, b:
node IDs

Returns:
--------
lca: [node 1, ..., node n]
list of lowest common ancestor nodes (can be more than one)
"""

assert nx.is_directed_acyclic_graph(graph), "Graph has to be acyclic and directed."

# get ancestors of both (intersection)
common_ancestors = list(nx.descendants(graph, a) & nx.descendants(graph, b))

# get sum of path lengths
sum_of_path_lengths = np.zeros((len(common_ancestors)))
for ii, c in enumerate(common_ancestors):
sum_of_path_lengths[ii] = nx.shortest_path_length(graph, a, c) \
+ nx.shortest_path_length(graph, b, c)

# print common_ancestors
# print sum_of_path_lengths

# return minima
minima, = np.where(sum_of_path_lengths == np.min(sum_of_path_lengths))

return [common_ancestors[ii] for ii in minima]

def test():

nodes = ["a","b","c","d","e","f","g","h","i","j","k","l","m","n","o","p"]
edges = [("a","b"),
("b","c"),
("b","d"),
("a","e"),
("a","h"),
("e","f"),
("e","g"),
("e","i"),
("h","l"),
("h","m"),
("g","j"),
("o","p"),
("o","n"),
("n","m"),
("n","l"),
("n","k"),
("p","j"),]

G = nx.DiGraph()
G.add_nodes_from(nodes)
G.add_edges_from(edges)

# plot
pos = nx.spring_layout(G)
nx.draw(G, pos)
nx.draw_networkx_labels(G, pos, labels=dict([(c, c) for c in 'abcdefghijklmnop']))
plt.show()

a,b = 'a','o'
lca = find_lowest_common_ancestor(G, a, b)
print "Lowest common ancestor(s) for {} and {}: {}".format(a, b, lca)

return

if __name__ == "__main__":
test()

关于python - NetworkX 的所有最低共同祖先,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/39946894/

35 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com