-
Notifications
You must be signed in to change notification settings - Fork 0
/
example_2.mod
58 lines (46 loc) · 1.64 KB
/
example_2.mod
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
# Traveling Salesman Problem (incorrect one because of lack of sub-tour removal)
set PLACES;
param START symbolic, in PLACES;
param lat{PLACES};
param lng{PLACES};
param N := card(PLACES);
# compute great circle distances and minimum travel times
param d2r := 3.1415926/180;
param alpha{a in PLACES, b in PLACES} := sin(d2r*(lat[a]-lat[b])/2)**2
+ cos(d2r*lat[a])*cos(d2r*lat[b])*sin(d2r*(lng[a]-lng[b])/2)**2;
param gcdist{a in PLACES, b in PLACES} := 2*6371*atan(sqrt(alpha[a,b]),sqrt(1-alpha[a,b]));
# Path constraints
var x{PLACES, PLACES} binary;
var y{PLACES, PLACES} >= 0 integer;
# must leave from all places
s.t. lv1 {a in PLACES}: sum{b in PLACES} x[a,b] = 1;
# must arrive at all places
s.t. lv2 {a in PLACES}: sum{b in PLACES} x[b,a] = 1;
# flow
s.t. flow1 {a in PLACES, b in PLACES}: y[a,b] <= (N - 1) * x[a,b];
s.t. flow2 {a in PLACES}: sum{b in PLACES} y[b,a] + (if a = START then N) = sum{b in PLACES} y[a,b] + 1;
minimize obj: sum{a in PLACES} (sum{b in PLACES} (x[a,b] * gcdist[a,b]));
solve;
printf "\nTour:\n\n";
for {a in PLACES} {
for {b in PLACES} {
# GLPK has no IF ... THEN ... ELSE for expressions
for {{0}: x[a,b]}{
printf "%3s -> %3s\n", a, b;
}
}
}
printf "\nTotal tour distance: %f\n\n", obj;
data;
param : PLACES : lat lng :=
ATL 33.6366995 -84.4278639
BOS 42.3629722 -71.0064167
DEN 39.8616667 -104.6731667
DFW 32.8968281 -97.0379958
JFK 40.6397511 -73.7789256
LAX 33.9424955 -118.4080684
ORD 41.9816486 -87.9066714
STL 38.7486972 -90.3700289
;
param START ATL;
end;