-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathtsp_solver.py
179 lines (153 loc) · 7.05 KB
/
tsp_solver.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
from itertools import islice
from array import array as pyarray
################################################################################
# A simple algorithm for solving the Travelling Salesman Problem
# Finds a suboptimal solution
################################################################################
def optimize_solution( distances, connections ):
"""Tries to optimize solution, found by the greedy algorithm"""
N = len(connections)
path = restore_path( connections )
def ds(i,j): #distance between ith and jth points of path
pi = path[i]
pj = path[j]
if pi < pj:
return distances[pj][pi]
else:
return distances[pi][pj]
d_total = 0.0
optimizations = 0
for a in range(N-1):
b = a+1
for c in range( b+2, N-1):
d = c+1
delta_d = ds(a,b)+ds(c,d) -( ds(a,c)+ds(b,d))
if delta_d > 0:
d_total += delta_d
optimizations += 1
connections[path[a]].remove(path[b])
connections[path[a]].append(path[c])
connections[path[b]].remove(path[a])
connections[path[b]].append(path[d])
connections[path[c]].remove(path[d])
connections[path[c]].append(path[a])
connections[path[d]].remove(path[c])
connections[path[d]].append(path[b])
path[:] = restore_path( connections )
return optimizations, d_total
def restore_path( connections ):
"""Takes array of connections and returns a path.
Connections is array of lists with 1 or 2 elements.
These elements are indices of teh vertices, connected to this vertex
Guarantees that first index < last index
"""
global startpoint, finishpoint
if startpoint is None:
#there are 2 nodes with valency 1 - start and end. Get them.
for idx, conn in enumerate(connections):
if len(conn)==1 and idx != finishpoint:
startpoint = idx
path = [startpoint]
prev_point = None
cur_point = startpoint
while True:
next_points = [pnt for pnt in connections[cur_point]
if pnt != prev_point ]
if not next_points: break
next_point = next_points[0]
path.append(next_point)
prev_point, cur_point = cur_point, next_point
return path
def _assert_triangular(distances):
"""Ensure that matrix is left-triangular at least.
"""
for i, row in enumerate(distances):
if len(row) < i: raise ValueError( "Distance matrix must be left-triangular at least. Row {row} must have at least {i} items".format(**locals()))
def pairs_by_dist(N, distances):
"""returns list of coordinate pairs (i,j), sorted by distances; such that i < j"""
#Sort coordinate pairs by distance
indices = []
for i in range(N):
for j in range(i):
indices.append(i*N+j)
indices.sort(key = lambda ij: distances[ij//N][ij%N])
return ((ij//N,ij%N) for ij in indices)
def solve_tsp( distances, optim_steps=3, pairs_by_dist=pairs_by_dist, startpt=None, finishpt=None ):
"""Given a distance matrix, finds a solution for the TSP problem.
Returns list of vertex indices.
Guarantees that the first index is lower than the last
:arg: distances : left-triangular matrix of distances. array of arrays
:arg: optim_steps (int) number of additional optimization steps, allows to improve solution but costly.
:arg: pairs_by_dist (function) an implementtion of the pairs_by_dist function. for optimization purposes.
:arg: startpoint : None or index that you want as the starting point (int)
:arg: finishpoint : None or index that you want as the finishing point (int)
"""
global startpoint, finishpoint
startpoint, finishpoint = startpt, finishpt
N = len(distances)
if N == 0: return []
if N == 1: return [0]
_assert_triangular(distances)
#State of the TSP solver algorithm.
node_valency = pyarray('i', [2])*N #Initially, each node has 2 sticky ends
if startpoint is not None:
node_valency[startpoint]=1
if finishpoint is not None:
node_valency[finishpoint]=1
if (startpoint is not None or finishpoint is not None) and startpoint == finishpoint:
print("The starting and finishing points cannot be the same while using the greedy algorithm")
raise ValueError("start=end is not supported")
#for each node, stores 1 or 2 connected nodes
connections = [[] for i in range(N)]
def join_segments(sorted_pairs):
#segments of nodes. Initially, each segment contains only 1 node
segments = [ [i] for i in range(N) ]
def possible_edges():
#Generate sequence of graph edges, that are possible and connect different segments.
#print("#### sorted pairs:", sorted_pairs)
for ij in sorted_pairs:
i,j = ij
#if both start and end could have connections,
# and both nodes connect to a different segments:
if node_valency[i] and node_valency[j] and\
(segments[i] is not segments[j]):
yield ij
def connect_vertices(i,j):
node_valency[i] -= 1
node_valency[j] -= 1
connections[i].append(j)
connections[j].append(i)
#Merge segment J into segment I.
seg_i = segments[i]
seg_j = segments[j]
if len(seg_j) > len(seg_i):
seg_i, seg_j = seg_j, seg_i
i, j = j, i
for node_idx in seg_j:
segments[node_idx] = seg_i
seg_i.extend(seg_j)
def edge_connects_endpoint_segments(i,j):
#return True, if given ede merges 2 segments that have endpoints in them
si,sj = segments[i],segments[j]
ss,se = segments[startpoint], segments[finishpoint]
return (si is ss) and (sj is se) or (sj is ss) and (si is se)
#Take first N-1 possible edge. they are already sorted by distance
edges_left = N-1
for i,j in possible_edges():
if (startpoint is not None and finishpoint is not None) and edges_left!=1 and edge_connects_endpoint_segments(i,j):
#print(f"#### disallow {i}, {j} because premature termination")
continue #don't allow premature path termination
#print(f"####add edge {i}, {j} of len {distances[i][j]}")
connect_vertices(i,j)
edges_left -= 1
if edges_left == 0:
break
#invoke main greedy algorithm
join_segments(pairs_by_dist(N, distances))
#now call additional optiomization procedure.
for passn in range(optim_steps):
nopt, dtotal = optimize_solution( distances, connections )
if nopt == 0:
break
#restore path from the connections map (graph) and return it
return restore_path( connections )