import math
import pulp
# Haversine formula to calculate distance between two points on the Earth
def haversine(coord1, coord2):
= 6371.0 # Earth radius in kilometers
R
= coord1
lat1, lon1 = coord2
lat2, lon2
= math.radians(lat2 - lat1)
dlat = math.radians(lon2 - lon1)
dlon
= math.sin(dlat / 2) ** 2 + math.cos(math.radians(lat1)) * math.cos(math.radians(lat2)) * math.sin(dlon / 2) ** 2
a = 2 * math.atan2(math.sqrt(a), math.sqrt(1 - a))
c
= R * c
distance return distance
# Define the problem
= pulp.LpProblem("p-median", pulp.LpMinimize)
prob
# Example data
# Coordinates of demand points and candidate facility locations
= [(37.4907391344, 127.0154235196), (37.4917391344, 127.0164235196)] # Example coordinates
demand_points = [(37.4927391344, 127.0174235196), (37.4937391344, 127.0184235196)]
candidate_facilities
# Demand quantities for each demand point
= [100, 150] # Example demand quantities
demand_quantities
= 1 # Number of facilities to select
p
# Define decision variables
= pulp.LpVariable.dicts("facility", range(len(candidate_facilities)), cat='Binary')
x = pulp.LpVariable.dicts("assign", [(i, j) for i in range(len(demand_points)) for j in range(len(candidate_facilities))], cat='Binary')
y
# Define the objective function
+= pulp.lpSum(
prob
pulp.lpSum(* demand_quantities[i] * haversine(demand_points[i], candidate_facilities[j])
y[(i, j)] for j in range(len(candidate_facilities))
) for i in range(len(demand_points))
)
# Define constraints
for i in range(len(demand_points)):
+= pulp.lpSum(y[(i, j)] for j in range(len(candidate_facilities))) == 1
prob += pulp.lpSum(x[j] for j in range(len(candidate_facilities))) == p
prob for i in range(len(demand_points)):
for j in range(len(candidate_facilities)):
+= y[(i, j)] <= x[j]
prob
# Solve the problem
prob.solve()
# Output the results
for v in prob.variables():
print(v.name, "=", v.varValue)
Welcome to the CBC MILP Solver
Version: 2.10.3
Build Date: Dec 15 2019
command line - /opt/anaconda3/lib/python3.12/site-packages/pulp/solverdir/cbc/osx/64/cbc /var/folders/5_/ym1r3my96mb6yq3bdn1242bw0000gp/T/ae306e2e5033456f8aa6cccddd1af83b-pulp.mps -timeMode elapsed -branch -printingOptions all -solution /var/folders/5_/ym1r3my96mb6yq3bdn1242bw0000gp/T/ae306e2e5033456f8aa6cccddd1af83b-pulp.sol (default strategy 1)
At line 2 NAME MODEL
At line 3 ROWS
At line 12 COLUMNS
At line 43 RHS
At line 51 BOUNDS
At line 58 ENDATA
Problem MODEL has 7 rows, 6 columns and 14 elements
Coin0008I MODEL read with 0 errors
Option for timeMode changed from cpu to elapsed
Continuous objective value is 49.6805 - 0.00 seconds
Cgl0003I 0 fixed, 0 tightened bounds, 0 strengthened rows, 7 substitutions
Cgl0004I processed model has 0 rows, 0 columns (0 integer (0 of which binary)) and 0 elements
Cbc3007W No integer variables - nothing to do
Cuts at root node changed objective from 49.6805 to -1.79769e+308
Probing was tried 0 times and created 0 cuts of which 0 were active after adding rounds of cuts (0.000 seconds)
Gomory was tried 0 times and created 0 cuts of which 0 were active after adding rounds of cuts (0.000 seconds)
Knapsack was tried 0 times and created 0 cuts of which 0 were active after adding rounds of cuts (0.000 seconds)
Clique was tried 0 times and created 0 cuts of which 0 were active after adding rounds of cuts (0.000 seconds)
MixedIntegerRounding2 was tried 0 times and created 0 cuts of which 0 were active after adding rounds of cuts (0.000 seconds)
FlowCover was tried 0 times and created 0 cuts of which 0 were active after adding rounds of cuts (0.000 seconds)
TwoMirCuts was tried 0 times and created 0 cuts of which 0 were active after adding rounds of cuts (0.000 seconds)
ZeroHalf was tried 0 times and created 0 cuts of which 0 were active after adding rounds of cuts (0.000 seconds)
Result - Optimal solution found
Objective value: 49.68052983assign_(0,_0) = 1.0
assign_(0,_1) = 0.0
assign_(1,_0) = 1.0
assign_(1,_1) = 0.0
facility_0 = 1.0
facility_1 = 0.0