找到两个给定节点之间的路径?
假设我有以下方式连接的节点,如何获得给定点之间存在的路径数量以及路径详细信息?
1,2 //node 1 and 2 are connected
2,3
2,5
4,2
5,11
11,12
6,7
5,6
3,6
6,8
8,10
8,9
求从1到7的路径:
答案: 找到 2 个路径,它们是
1,2,3,6,7
1,2,5,6,7
找到的实现 这里 很好,我将使用相同的
这里是上面链接中 python 的片段
# a sample graph
graph = {'A': ['B', 'C','E'],
'B': ['A','C', 'D'],
'C': ['D'],
'D': ['C'],
'E': ['F','D'],
'F': ['C']}
class MyQUEUE: # just an implementation of a queue
def __init__(self):
self.holder = []
def enqueue(self,val):
self.holder.append(val)
def dequeue(self):
val = None
try:
val = self.holder[0]
if len(self.holder) == 1:
self.holder = []
else:
self.holder = self.holder[1:]
except:
pass
return val
def IsEmpty(self):
result = False
if len(self.holder) == 0:
result = True
return result
path_queue = MyQUEUE() # now we make a queue
def BFS(graph,start,end,q):
temp_path = [start]
q.enqueue(temp_path)
while q.IsEmpty() == False:
tmp_path = q.dequeue()
last_node = tmp_path[len(tmp_path)-1]
print tmp_path
if last_node == end:
print "VALID_PATH : ",tmp_path
for link_node in graph[last_node]:
if link_node not in tmp_path:
#new_path = []
new_path = tmp_path + [link_node]
q.enqueue(new_path)
BFS(graph,"A","D",path_queue)
-------------results-------------------
['A']
['A', 'B']
['A', 'C']
['A', 'E']
['A', 'B', 'C']
['A', 'B', 'D']
VALID_PATH : ['A', 'B', 'D']
['A', 'C', 'D']
VALID_PATH : ['A', 'C', 'D']
['A', 'E', 'F']
['A', 'E', 'D']
VALID_PATH : ['A', 'E', 'D']
['A', 'B', 'C', 'D']
VALID_PATH : ['A', 'B', 'C', 'D']
['A', 'E', 'F', 'C']
['A', 'E', 'F', 'C', 'D']
VALID_PATH : ['A', 'E', 'F', 'C', 'D']
如果你对这篇内容有疑问,欢迎到本站社区发帖提问 参与讨论,获取更多帮助,或者扫码二维码加入 Web 技术交流群。
绑定邮箱获取回复消息
由于您还没有绑定你的真实邮箱,如果其他用户或者作者回复了您的评论,将不能在第一时间通知您!
发布评论
评论(8)
广度优先搜索遍历一个图,实际上找到从起始节点开始的所有路径。 然而,BFS 通常不会保留所有路径。 相反,它更新前驱函数 π 以保存最短路径。 您可以轻松修改算法,以便
π(n)
不仅存储一个 前驱,还存储可能的前驱列表。然后所有可能的路径都被编码在这个函数中,通过递归遍历 π 就可以得到所有可能的路径组合。
Cormen 等人在算法简介中可以找到使用这种表示法的一个很好的伪代码,并且随后在许多大学关于该主题的脚本中使用。 通过 Google 搜索“BFS 伪代码前身 π”,Stack Exchange 上的这一搜索结果被连根拔起。
Breadth-first search traverses a graph and in fact finds all paths from a starting node. Usually, BFS doesn't keep all paths, however. Instead, it updates a prededecessor function π to save the shortest path. You can easily modify the algorithm so that
π(n)
doesn't only store one predecessor but a list of possible predecessors.Then all possible paths are encoded in this function, and by traversing π recursively you get all possible path combinations.
One good pseudocode which uses this notation can be found in Introduction to Algorithms by Cormen et al. and has subsequently been used in many University scripts on the subject. A Google search for “BFS pseudocode predecessor π” uproots this hit on Stack Exchange.
对于那些不是 PYTHON 专家的人,C++ 中的相同代码
For those who are not PYTHON expert ,the same code in C++
Dijkstra 的算法更多地适用于加权路径,听起来发帖者想要找到所有路径,而不仅仅是最短的路径。
对于此应用程序,我将构建一个图表(您的应用程序听起来不需要定向)并使用您最喜欢的搜索方法。 听起来您想要所有路径,而不仅仅是猜测最短的路径,因此请使用您选择的简单递归算法。
唯一的问题是图是否可以是循环的。
连接:
在寻找从 1->4 的路径时,可能会出现 1->4 的循环。 2-> 3-> 1.
在这种情况下,我会在遍历节点时保留一个堆栈。 以下是该图的步骤和生成的堆栈的列表(抱歉格式 - 没有表格选项):
当前节点(可能的下一个节点减去我们来自的位置)[堆栈]
Dijkstra's algorithm applies more to weighted paths and it sounds like the poster was wanting to find all paths, not just the shortest.
For this application, I'd build a graph (your application sounds like it wouldn't need to be directed) and use your favorite search method. It sounds like you want all paths, not just a guess at the shortest one, so use a simple recursive algorithm of your choice.
The only problem with this is if the graph can be cyclic.
With the connections:
While looking for a path from 1->4, you could have a cycle of 1 -> 2 -> 3 -> 1.
In that case, then I'd keep a stack as traversing the nodes. Here's a list with the steps for that graph and the resulting stack (sorry for the formatting - no table option):
current node (possible next nodes minus where we came from) [stack]
原始代码有点麻烦,如果您想使用 BFS 查找图上两点之间是否存在路径,您可能需要使用 collections.deque。 这是我编写的一个快速解决方案:
注意:如果两个节点之间不存在路径,则此方法可能会无限继续。 我还没有测试所有的情况,YMMV。
The original code is a bit cumbersome and you might want to use the collections.deque instead if you want to use BFS to find if a path exists between 2 points on the graph. Here is a quick solution I hacked up:
Note: this method might continue infinitely if there exists no path between the two nodes. I haven't tested all cases, YMMV.
在 Prolog(具体来说,SWI-Prolog)
测试中:
In Prolog (specifically, SWI-Prolog)
test:
如果需要所有路径,请使用递归。
最好使用邻接列表创建一个函数 f() 来尝试填充当前访问过的顶点列表。 像这样:
由于向量是按值传递的(因此在递归过程中进一步进行的任何更改都不是永久性的),因此会枚举所有可能的组合。
您可以通过引用传递前一个向量来提高效率(因此不需要一遍又一遍地复制向量),但您必须确保手动执行 popped_back() 。
还有一件事:如果图表有循环,这将不起作用。
(我假设在这种情况下,您需要找到所有简单路径,然后)在将某些内容添加到前一个向量中之前,首先检查它是否已经存在于其中。
如果您想要所有最短路径,请使用 Konrad 对此算法的建议。
If you want all the paths, use recursion.
Using an adjacency list, preferably, create a function f() that attempts to fill in a current list of visited vertices. Like so:
Due to the fact that the vector is passed by value (and thus any changes made further down in the recursive procedure aren't permanent), all possible combinations are enumerated.
You can gain a bit of efficiency by passing the previous vector by reference (and thus not needing to copy the vector over and over again) but you'll have to make sure that things get popped_back() manually.
One more thing: if the graph has cycles, this won't work.
(I assume in this case you'll want to find all simple paths, then) Before adding something into the previous vector, first check if it's already in there.
If you want all shortest paths, use Konrad's suggestion with this algorithm.
给定邻接矩阵:
{0, 1, 3, 4, 0, 0}
{0, 0, 2, 1, 2, 0}
{0, 1, 0, 3, 0, 0}
{0, 1, 1 , 0, 0, 1}
{0, 0, 0, 0, 0, 6}
{0, 1, 0, 1, 0, 0}
以下 Wolfram Mathematica 代码解决了查找两个节点之间所有简单路径的问题的图表。
我使用简单的递归和两个全局变量来跟踪循环并存储所需的输出。
代码并没有仅仅为了代码的清晰度而进行优化。
“打印”应该有助于阐明它是如何工作的。
调用代码:
初始化节点=1;
结束节点 = 6;
循环={};
树= {{initNode}};
builtTree[initNode, 矩阵];
路径:{{1}}
root:1---节点:{2,3,4}
路径: {{1,2},{1,3},{1,4}}
root:2---节点:{3,4,5}
路径:{{1,3},{1,4},{1,2,3},{1,2,4},{1,2 ,5}}
根:3---节点:{2,4}
路径:{{1,4},{1,2,4},{1,2,5},{1,3,4},{1,2 ,3,4},{1,3,2,4},{1,3,2,5}}
root:4---节点:{2,3,6}
路径:{{1,2,5},{1,3,2,5},{1,4,6},{1,2,4 ,6},{1,3,4,6},{1,2,3,4,6},{1,3,2,4,6},{1,4,2,5},{1 ,3,4,2,5},{1,4,3,2,5}}
根:5---节点:{6}
结果:{{1, 4, 6}, {1, 2, 4, 6}, {1, 2, 5, 6}, {1, 3, 4, 6 }, {1, 2, 3, 4, 6}, {1, 3, 2, 4, 6}, {1, 3, 2, 5, 6}, {1, 4, 2, 5, 6}, {1, 3, 4, 2, 5, 6}, {1, 4, 3, 2, 5, 6}}
...不幸的是,我无法上传图像以更好的方式显示结果:(
http://textanddatamining.blogspot.com
given the adjacency matrix:
{0, 1, 3, 4, 0, 0}
{0, 0, 2, 1, 2, 0}
{0, 1, 0, 3, 0, 0}
{0, 1, 1, 0, 0, 1}
{0, 0, 0, 0, 0, 6}
{0, 1, 0, 1, 0, 0}
the following Wolfram Mathematica code solve the problem to find all the simple paths between two nodes of a graph.
I used simple recursion, and two global var to keep track of cycles and to store the desired output.
the code hasn't been optimized just for the sake of code clarity.
the "print" should be helpful to clarify how it works.
to call the code:
initNode = 1;
endNode = 6;
lcycle = {};
tree = {{initNode}};
builtTree[initNode, matrix];
paths: {{1}}
root:1---nodes:{2,3,4}
paths: {{1,2},{1,3},{1,4}}
root:2---nodes:{3,4,5}
paths: {{1,3},{1,4},{1,2,3},{1,2,4},{1,2,5}}
root:3---nodes:{2,4}
paths: {{1,4},{1,2,4},{1,2,5},{1,3,4},{1,2,3,4},{1,3,2,4},{1,3,2,5}}
root:4---nodes:{2,3,6}
paths: {{1,2,5},{1,3,2,5},{1,4,6},{1,2,4,6},{1,3,4,6},{1,2,3,4,6},{1,3,2,4,6},{1,4,2,5},{1,3,4,2,5},{1,4,3,2,5}}
root:5---nodes:{6}
RESULTS:{{1, 4, 6}, {1, 2, 4, 6}, {1, 2, 5, 6}, {1, 3, 4, 6}, {1, 2, 3, 4, 6}, {1, 3, 2, 4, 6}, {1, 3, 2, 5, 6}, {1, 4, 2, 5, 6}, {1, 3, 4, 2, 5, 6}, {1, 4, 3, 2, 5, 6}}
...Unfortunately I cannot upload images to show the results in a better way :(
http://textanddatamining.blogspot.com
您要做的本质上是在(有向?)图中找到两个顶点之间的路径,请查看 Dijkstra 算法,如果您需要最短路径,或者编写一个简单的递归函数,如果您需要任何存在的路径。
What you're trying to do is essentially to find a path between two vertices in a (directed?) graph check out Dijkstra's algorithm if you need shortest path or write a simple recursive function if you need whatever paths exist.