forked from google/or-tools
-
Notifications
You must be signed in to change notification settings - Fork 9
/
Copy pathbin_packing_sat.py
146 lines (122 loc) · 3.94 KB
/
bin_packing_sat.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
#!/usr/bin/env python3
# Copyright 2010-2024 Google LLC
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
# http://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""Solves a simple bin packing problem using CP-SAT."""
# [START program]
# [START import]
import io
import pandas as pd
from ortools.sat.python import cp_model
# [END import]
# [START program_part1]
# [START data_model]
def create_data_model() -> tuple[pd.DataFrame, pd.DataFrame]:
"""Create the data for the example."""
items_str = """
item weight
i1 48
i2 30
i3 19
i4 36
i5 36
i6 27
i7 42
i8 42
i9 36
i10 24
i11 30
"""
bins_str = """
bin capacity
b1 100
b2 100
b3 100
b4 100
b5 100
b6 100
b7 100
"""
items = pd.read_table(io.StringIO(items_str), index_col=0, sep=r"\s+")
bins = pd.read_table(io.StringIO(bins_str), index_col=0, sep=r"\s+")
return items, bins
# [END data_model]
def main() -> None:
# [START data]
items, bins = create_data_model()
# [END data]
# [END program_part1]
# [START model]
# Create the model.
model = cp_model.CpModel()
# [END model]
# [START program_part2]
# [START variables]
# Variables
# x[i, j] = 1 if item i is packed in bin j.
items_x_bins = pd.MultiIndex.from_product(
[items.index, bins.index], names=["item", "bin"]
)
x = model.new_bool_var_series(name="x", index=items_x_bins)
# y[j] = 1 if bin j is used.
y = model.new_bool_var_series(name="y", index=bins.index)
# [END variables]
# [START constraints]
# Constraints
# Each item must be in exactly one bin.
for unused_name, all_copies in x.groupby("item"):
model.add_exactly_one(x[all_copies.index])
# The amount packed in each bin cannot exceed its capacity.
for selected_bin in bins.index:
items_in_bin = x.xs(selected_bin, level="bin")
model.add(
items_in_bin.dot(items.weight)
<= bins.loc[selected_bin].capacity * y[selected_bin]
)
# [END constraints]
# [START objective]
# Objective: minimize the number of bins used.
model.minimize(y.sum())
# [END objective]
# [START solve]
# Create the solver and solve the model.
solver = cp_model.CpSolver()
status = solver.solve(model)
# [END solve]
# [START print_solution]
if status == cp_model.OPTIMAL or status == cp_model.FEASIBLE:
print(f"Number of bins used = {solver.objective_value}")
x_values = solver.boolean_values(x)
y_values = solver.boolean_values(y)
active_bins = y_values.loc[lambda x: x].index
for b in active_bins:
print(f"Bin {b}")
items_in_active_bin = x_values.xs(b, level="bin").loc[lambda x: x].index
for item in items_in_active_bin:
print(f" Item {item} - weight {items.loc[item].weight}")
print(
" Packed items weight:"
f" {items.loc[items_in_active_bin].sum().to_string()}"
)
print()
print(f"Total packed weight: {items.weight.sum()}")
print()
print(f"Time = {solver.wall_time} seconds")
elif status == cp_model.INFEASIBLE:
print("No solution found")
else:
print("Something is wrong, check the status and the log of the solve")
# [END print_solution]
if __name__ == "__main__":
main()
# [END program_part2]
# [END program]