forked from google/or-tools
-
Notifications
You must be signed in to change notification settings - Fork 0
/
linear_constraint_manager.h
161 lines (132 loc) · 6.16 KB
/
linear_constraint_manager.h
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
// Copyright 2010-2018 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.
#ifndef OR_TOOLS_SAT_LINEAR_CONSTRAINT_MANAGER_H_
#define OR_TOOLS_SAT_LINEAR_CONSTRAINT_MANAGER_H_
#include <cstddef>
#include <vector>
#include "absl/container/flat_hash_map.h"
#include "absl/container/flat_hash_set.h"
#include "ortools/sat/linear_constraint.h"
#include "ortools/sat/model.h"
#include "ortools/sat/sat_parameters.pb.h"
namespace operations_research {
namespace sat {
// This class holds a list of globally valid linear constraints and has some
// logic to decide which one should be part of the LP relaxation. We want more
// for a better relaxation, but for efficiency we do not want to have too much
// constraints while solving the LP.
//
// This class is meant to contain all the initial constraints of the LP
// relaxation and to get new cuts as they are generated. Thus, it can both
// manage cuts but also only add the initial constraints lazily if there is too
// many of them.
//
// TODO(user): Also store the LP objective there as it can be useful to decide
// which constraint should go into the current LP.
class LinearConstraintManager {
public:
struct ConstraintInfo {
LinearConstraint constraint;
double l2_norm;
bool is_cut;
int64 inactive_count;
double objective_parallelism;
bool objective_parallelism_computed;
bool is_in_lp;
// A constraint might be marked for permanent removal if it is almost
// parallel to one of the existing constraints in the LP.
bool permanently_removed;
size_t hash;
};
explicit LinearConstraintManager(Model* model)
: sat_parameters_(*model->GetOrCreate<SatParameters>()),
integer_trail_(*model->GetOrCreate<IntegerTrail>()) {}
~LinearConstraintManager();
// Add a new constraint to the manager. Note that we canonicalize constraints
// and merge the bounds of constraints with the same terms. We also perform
// basic preprocessing.
DEFINE_INT_TYPE(ConstraintIndex, int32);
ConstraintIndex Add(LinearConstraint ct);
// Same as Add(), but logs some information about the newly added constraint.
// Cuts are also handled slightly differently than normal constraints.
void AddCut(LinearConstraint ct, std::string type_name,
const gtl::ITIVector<IntegerVariable, double>& lp_solution);
// The objective is used as one of the criterion to score cuts.
// The more a cut is parallel to the objective, the better its score is.
void SetObjectiveCoefficient(IntegerVariable var, IntegerValue coeff);
// Heuristic to decides what LP is best solved next. The given lp_solution
// should usually be the optimal solution of the LP returned by GetLp() before
// this call, but is just used as an heuristic.
//
// Returns true iff LpConstraints() will return a different LP than before.
bool ChangeLp(const gtl::ITIVector<IntegerVariable, double>& lp_solution);
// This can be called initially to add all the current constraint to the LP
// returned by GetLp().
void AddAllConstraintsToLp();
// All the constraints managed by this class.
const gtl::ITIVector<ConstraintIndex, ConstraintInfo>& AllConstraints()
const {
return constraint_infos_;
}
// The set of constraints indices in AllConstraints() that should be part
// of the next LP to solve.
const std::vector<ConstraintIndex>& LpConstraints() const {
return lp_constraints_;
}
int64 num_cuts() const { return num_cuts_; }
int64 num_shortened_constraints() const { return num_shortened_constraints_; }
int64 num_coeff_strenghtening() const { return num_coeff_strenghtening_; }
private:
// Removes the marked constraints from the LP.
void RemoveMarkedConstraints();
// Apply basic inprocessing simplification rules:
// - remove fixed variable
// - reduce large coefficient (i.e. coeff strenghtenning or big-M reduction).
// This uses level-zero bounds.
// Returns true if the terms of the constraint changed.
bool SimplifyConstraint(LinearConstraint* ct);
// Helper method to compute objective parallelism for a given constraint. This
// also lazily computes objective norm.
void ComputeObjectiveParallelism(const ConstraintIndex ct_index);
const SatParameters& sat_parameters_;
const IntegerTrail& integer_trail_;
// Set at true by Add()/SimplifyConstraint() and at false by ChangeLp().
bool current_lp_is_changed_ = false;
// Optimization to avoid calling SimplifyConstraint() when not needed.
int64 last_simplification_timestamp_ = 0;
gtl::ITIVector<ConstraintIndex, ConstraintInfo> constraint_infos_;
// Temporary list of constraints marked for removal. Note that we remove
// constraints in batch to avoid changing LP too frequently.
absl::flat_hash_set<ConstraintIndex> constraints_removal_list_;
// The subset of constraints currently in the lp.
std::vector<ConstraintIndex> lp_constraints_;
// We keep a map from the hash of our constraint terms to their position in
// constraints_. This is an optimization to detect duplicate constraints. We
// are robust to collisions because we always relies on the ground truth
// contained in constraints_ and the code is still okay if we do not merge the
// constraints.
absl::flat_hash_map<size_t, ConstraintIndex> equiv_constraints_;
int64 num_merged_constraints_ = 0;
int64 num_shortened_constraints_ = 0;
int64 num_splitted_constraints_ = 0;
int64 num_coeff_strenghtening_ = 0;
int64 num_cuts_ = 0;
std::map<std::string, int> type_to_num_cuts_;
bool objective_is_defined_ = false;
bool objective_norm_computed_ = false;
LinearConstraint objective_;
double objective_l2_norm_ = 0.0;
};
} // namespace sat
} // namespace operations_research
#endif // OR_TOOLS_SAT_LINEAR_CONSTRAINT_MANAGER_H_