我有一个2D点列表,例如:
1,1 2,2 1,3 4,5 2,1
这些点之间的距离是已知的(例如,使用math.hypot。)我想对列表进行排序,以便它们之间存在最小距离。我可以使用任何可能的解决方案订单,只要这些点是最短的顺序。
实现这一目标的最佳方式是什么?
我正在考虑计算任何项目和任何其他项目之间的距离,并且每次选择最小的项目,但这对我正在处理的列表来说是一个缓慢的算法(1,000项不会不常见。)
答案 0 :(得分:7)
您要问的技术问题类似于“图形的minimum hamiltonian path是什么”(您的元组是顶点,它们之间的距离是边的权重)。这个问题不能在多项式时间内解决,因此您的数据集最好小。由于您的图表已完成(所有节点都已连接),因此最小哈密顿路径问题可能无法完全应用。
无论如何,下面的答案都使用蛮力。它会置换所有可能的路径,计算每条路径的距离,然后获得最小路径。
import itertools as it
import math
def dist(x,y):
return math.hypot(y[0]-x[0],y[1]-x[1])
paths = [ p for p in it.permutations([(1,2),(2,3),(5,6),(3,4)]) ]
path_distances = [ sum(map(lambda x: dist(x[0],x[1]),zip(p[:-1],p[1:]))) for p in paths ]
min_index = argmin(path_distances)
print paths[min_index], path_distances[min_index]
输出:
((1, 2), (2, 3), (3, 4), (5, 6)) 5.65685424949
请注意,反向路径是等效的最小值
答案 1 :(得分:2)
这里的另一个答案是正确的,这是一类NP问题。如果你真的需要1000个节点,那么你无法真正解决它。但它需要确切吗?如果没有,也许你可以尝试选择一个随机点,每次从那里走到最近的点?它不能保证给你最短的路径,但也许它足够接近。 E.g:
data [ (1,2), (3,4), ... ]
cur = 0
path = [cur]
totalDist = 0
for i in range(1,len(data)):
dists = [(dist(data[i],p), pi) for (pi,p) in enumerate(data) if pi != i and pi not in path]
nextDist, cur = min(dists)
totalDist += nextDist
path.append(cur)
print path, totalDist
这是距离计算和比较中的O(n ^ 2),内存中只有O(n),至少可以达到1000点。