-
Notifications
You must be signed in to change notification settings - Fork 30
/
hungarian_bigraph_matcher.hpp
247 lines (193 loc) · 8.94 KB
/
hungarian_bigraph_matcher.hpp
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
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
/******************************************************************************
* Copyright 2017 The Apollo Authors. All Rights Reserved.
*
* 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.
*****************************************************************************/
/*****************************************************************************
* Copyright 2010-2012 Google
*
* 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.
* See: //depot/or-tools/java/com/google/wireless/genie/frontend
* /mixer/matching/HungarianOptimizer.java
****************************************************************************/
#ifndef COMMON_INCLUDE_COMMON_ALGOS_HUNGARIAN_BIGRAPH_MATCHER_HPP_
#define COMMON_INCLUDE_COMMON_ALGOS_HUNGARIAN_BIGRAPH_MATCHER_HPP_
#include <vector>
namespace autosense {
namespace common {
namespace algos {
class HungarianBigraphMatcher {
static const int kRowNotFound = -1;
static const int kColNotFound = -2;
public:
// Setup the initial conditions for the algorithm.
// Parameters: costs is a matrix of the cost of assigning each agent to
// each task. costs[i][j] is the cost of assigning agent i to task j.
// All the costs must be non-negative. This matrix does not have to
// be square (i.e. we can have different numbers of agents and tasks), but
// it
// must be regular (i.e. there must be the same number of entries in each
// row
// of the matrix).
explicit HungarianBigraphMatcher(
const std::vector<std::vector<double>> &costs);
// Find an assignment which maximizes the total cost.
// Returns the assignment in the two vectors passed as argument.
// agent[i] is assigned to task[i].
void maximize(std::vector<int> *agent, std::vector<int> *task);
// Find an assignment which minimizes the total cost.
// Returns the assignment in the two vectors passed as argument.
// agent[i] is assigned to task[i].
void minimize(std::vector<int> *agent, std::vector<int> *task);
private:
typedef void (HungarianBigraphMatcher::*Step)();
typedef enum { NONE, PRIME, STAR } Mark;
// Convert the final cost matrix into a set of assignments of agents ->
// tasks.
// Returns the assignment in the two vectors passed as argument, the same as
// Minimize and Maximize
void find_assignments(std::vector<int> *agent, std::vector<int> *task);
// Is the cell (row, col) starred?
bool is_starred(int row, int col) const { return marks_[row][col] == STAR; }
// Mark cell (row, col) with a star
void star(int row, int col) {
marks_[row][col] = STAR;
stars_in_col_[col]++;
}
// Remove a star from cell (row, col)
void unstar(int row, int col) {
marks_[row][col] = NONE;
stars_in_col_[col]--;
}
// Find a column in row 'row' containing a star, or return
// kHungarianOptimizerColNotFound if no such column exists.
int find_star_in_row(int row) const;
// Find a row in column 'col' containing a star, or return
// kHungarianOptimizerRowNotFound if no such row exists.
int find_star_in_col(int col) const;
// Is cell (row, col) marked with a prime?
bool is_primed(int row, int col) const { return marks_[row][col] == PRIME; }
// Mark cell (row, col) with a prime.
void prime(int row, int col) { marks_[row][col] = PRIME; }
// Find a column in row containing a prime, or return
// kHungarianOptimizerColNotFound if no such column exists.
int find_prime_in_row(int row) const;
// Remove the prime marks_ from every cell in the matrix.
void clear_primes();
// Does column col contain a star?
bool col_contains_star(int col) const { return stars_in_col_[col] > 0; }
// Is row 'row' covered?
bool row_covered(int row) const { return rows_covered_[row]; }
// Cover row 'row'.
void cover_row(int row) { rows_covered_[row] = true; }
// Uncover row 'row'.
void uncover_row(int row) { rows_covered_[row] = false; }
// Is column col covered?
bool col_covered(int col) const { return cols_covered_[col]; }
// Cover column col.
void cover_col(int col) { cols_covered_[col] = true; }
// Uncover column col.
void uncover_col(int col) { cols_covered_[col] = false; }
// Uncover ever row and column in the matrix.
void clear_covers();
// Find the smallest uncovered cell in the matrix.
double find_smallest_uncovered();
// Find an uncovered zero and store its coordinates in (zeroRow_, zeroCol_)
// and return true, or return false if no such cell exists.
bool find_zero(int *zero_row, int *zero_col);
// Print the matrix to stdout (for debugging.)
void print_matrix();
// Run the Munkres algorithm!
void do_munkres();
void check_star();
// Step 1.
// For each row of the matrix, find the smallest element and subtract it
// from every element in its row. Go to Step 2.
void reduce_rows();
// Step 2.
// Find a zero (Z) in the matrix. If there is no starred zero in its row
// or column, star Z. Repeat for every element in the matrix. Go to step
// 3.
// Note: profiling shows this method to use 9.2% of the CPU - the next
// slowest step takes 0.6%. I can't think of a way of speeding it up
// though.
void star_zeroes();
// Step 3.
// Cover each column containing a starred zero. If all columns are
// covered, the starred zeros describe a complete set of unique assignments.
// In this case, terminate the algorithm. Otherwise, go to step 4.
void cover_starred_zeroes();
// Step 4.
// Find a noncovered zero and prime it. If there is no starred zero in the
// row containing this primed zero, Go to Step 5. Otherwise, cover this row
// and uncover the column containing the starred zero. Continue in this
// manner
// until there are no uncovered zeros left, then go to Step 6.
void prime_zeroes();
// Step 5.
// Construct a series of alternating primed and starred zeros as follows.
// Let Z0 represent the uncovered primed zero found in Step 4. Let Z1
// denote
// the starred zero in the column of Z0 (if any). Let Z2 denote the primed
// zero in the row of Z1 (there will always be one). Continue until the
// series terminates at a primed zero that has no starred zero in its
// column.
// Unstar each starred zero of the series, star each primed zero of the
// series, erase all primes and uncover every line in the matrix. Return to
// Step 3.
void make_augmenting_path();
// Step 6.
// Add the smallest uncovered value in the matrix to every element of each
// covered row, and subtract it from every element of each uncovered column.
// Return to Step 4 without altering any stars, primes, or covered lines.
void augment_path();
// The size of the problem, i.e. std::max(#agents, #tasks).
int matrix_size_;
// The expanded cost matrix.
std::vector<std::vector<double>> costs_;
// The greatest cost in the initial cost matrix.
double max_cost_;
// Which rows and columns are currently covered.
std::vector<bool> rows_covered_;
std::vector<bool> cols_covered_;
// The marks_ (star/prime/none) on each element of the cost matrix.
std::vector<std::vector<Mark>> marks_;
// The number of stars in each column - used to speed up coverStarredZeroes.
std::vector<int> stars_in_col_;
// Representation of a path_ through the matrix - used in step 5.
std::vector<int> preimage_; // i.e. the agents
std::vector<int> image_; // i.e. the tasks
// The locations of a zero found in step 4.
int zero_col_;
int zero_row_;
// The width_ and height_ of the initial (non-expanded) cost matrix.
int width_;
int height_;
// The current state of the algorithm
HungarianBigraphMatcher::Step state_;
std::vector<int> uncov_col_;
std::vector<int> uncov_row_;
};
} // namespace algos
} // namespace common
} // namespace autosense
#endif // COMMON_INCLUDE_COMMON_ALGOS_HUNGARIAN_BIGRAPH_MATCHER_HPP_