-
Notifications
You must be signed in to change notification settings - Fork 9
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Add support for indicator constraints in MPSolver #132
base: main
Are you sure you want to change the base?
Changes from 7 commits
59bb4f2
f243c06
73b8e15
4de1bfc
534c91c
a0e5dfd
52547b1
888963d
71aeeb8
b4f91fb
d68a748
05be41f
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
|
@@ -1511,6 +1511,33 @@ MPConstraint* MPSolver::MakeRowConstraint(const LinearRange& range, | |
return constraint; | ||
} | ||
|
||
MPConstraint* MPSolver::MakeIndicatorConstraint( | ||
double lb, double ub, const std::string& name, | ||
const MPVariable* indicator_variable, bool indicator_value) { | ||
if (!indicator_variable->integer() || indicator_variable->lb() != 0 || | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. To be consistent with e.g Although the 2nd line is currently included in the 1st. DLOG_IF(DFATAL, !interface_->solver_->OwnsVariable(indicator_variable)) << indicator_variable;
if (indicator_variable == nullptr) return nullptr; There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. good catch! |
||
indicator_variable->ub() != 1) { | ||
LOG(ERROR) << "Variable " << indicator_variable->name() | ||
<< " is not boolean"; | ||
pet-mit marked this conversation as resolved.
Show resolved
Hide resolved
|
||
return nullptr; | ||
} | ||
const int constraint_index = NumConstraints(); | ||
MPConstraint* const constraint = | ||
new MPConstraint(constraint_index, lb, ub, name, interface_.get()); | ||
constraint->indicator_variable_ = indicator_variable; | ||
constraint->indicator_value_ = indicator_value; | ||
if (!interface_->AddIndicatorConstraint(constraint)) { | ||
LOG(ERROR) << "Solver doesn't support indicator constraints"; | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I don't think it's necessary to display a log message here, it's already done in the base version of virtual bool AddIndicatorConstraint(MPConstraint* const /*ct*/) {
LOG(ERROR) << "Solver doesn't support indicator constraints.";
return false;
} It would be displayed twice. Not terrible, but not great either. There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. OK, but then for interfaces that can sometimes handle indicator constraints, sometimes not (for example an interface that can handle MIP or LP problems), we'll have to log the issue before returning "false" |
||
return nullptr; | ||
} | ||
if (constraint_name_to_index_) { | ||
gtl::InsertOrDie(&*constraint_name_to_index_, constraint->name(), | ||
constraint_index); | ||
} | ||
constraints_.push_back(constraint); | ||
constraint_is_extracted_.push_back(false); | ||
return constraint; | ||
} | ||
|
||
int MPSolver::ComputeMaxConstraintSize(int min_constraint_index, | ||
int max_constraint_index) const { | ||
int max_constraint_size = 0; | ||
|
Original file line number | Diff line number | Diff line change | ||||||||||||
---|---|---|---|---|---|---|---|---|---|---|---|---|---|---|
|
@@ -443,6 +443,16 @@ class MPSolver { | |||||||||||||
MPConstraint* MakeRowConstraint(const LinearRange& range, | ||||||||||||||
const std::string& name); | ||||||||||||||
|
||||||||||||||
/// Creates a named indicator constraint with given bounds and given | ||||||||||||||
/// indicator variable. | ||||||||||||||
Comment on lines
+446
to
+447
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more.
Suggested change
|
||||||||||||||
/// The constraint is active if and only if *indicator_variable has value | ||||||||||||||
/// indicator_value | ||||||||||||||
/// (Only available for MILP problems) | ||||||||||||||
MPConstraint* MakeIndicatorConstraint(double lb, double ub, | ||||||||||||||
const std::string& name, | ||||||||||||||
const MPVariable* indicator_variable, | ||||||||||||||
bool indicator_value); | ||||||||||||||
|
||||||||||||||
/** | ||||||||||||||
* Returns the objective object. | ||||||||||||||
* | ||||||||||||||
|
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,79 @@ | ||
// 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. | ||
|
||
#include <fstream> | ||
|
||
#include "gtest/gtest.h" | ||
#include "ortools/base/init_google.h" | ||
#include "ortools/linear_solver/linear_solver.h" | ||
|
||
namespace operations_research { | ||
|
||
TEST(ScipInterface, IndicatorConstraint0) { | ||
MPSolver solver("SCIP", MPSolver::SCIP_MIXED_INTEGER_PROGRAMMING); | ||
// Maximize x <= 100 | ||
auto x = solver.MakeNumVar(0, 100, "x"); | ||
solver.MutableObjective()->SetMaximization(); | ||
solver.MutableObjective()->SetCoefficient(x, 1); | ||
// With indicator constraint | ||
// if var = 0, then x <= 10 | ||
auto var = solver.MakeBoolVar("indicator_var"); | ||
auto ct = solver.MakeIndicatorConstraint(0, 10, "test", var, false); | ||
ct->SetCoefficient(x, 1); | ||
|
||
// Leave var free ==> x = 100 | ||
solver.Solve(); | ||
EXPECT_EQ(var->solution_value(), 1); | ||
EXPECT_EQ(x->solution_value(), 100); | ||
|
||
// Force var to 0 ==> x = 10 | ||
var->SetUB(0); | ||
solver.Solve(); | ||
EXPECT_EQ(x->solution_value(), 10); | ||
} | ||
|
||
TEST(ScipInterface, IndicatorConstraint1) { | ||
MPSolver solver("SCIP", MPSolver::SCIP_MIXED_INTEGER_PROGRAMMING); | ||
// Maximize x <= 100 | ||
auto x = solver.MakeNumVar(0, 100, "x"); | ||
solver.MutableObjective()->SetMaximization(); | ||
solver.MutableObjective()->SetCoefficient(x, 1); | ||
// With indicator constraint | ||
// if var = 1, then x <= 10 | ||
auto var = solver.MakeBoolVar("indicator_var"); | ||
auto ct = solver.MakeIndicatorConstraint(0, 10, "test", var, true); | ||
ct->SetCoefficient(x, 1); | ||
|
||
// Leave var free ==> x = 100 | ||
solver.Solve(); | ||
EXPECT_EQ(var->solution_value(), 0); | ||
EXPECT_EQ(x->solution_value(), 100); | ||
|
||
// Force var to 0 ==> x = 10 | ||
var->SetLB(1); | ||
solver.Solve(); | ||
EXPECT_EQ(x->solution_value(), 10); | ||
} | ||
} // namespace operations_research | ||
|
||
int main(int argc, char** argv) { | ||
absl::SetFlag(&FLAGS_stderrthreshold, 0); | ||
testing::InitGoogleTest(&argc, argv); | ||
auto solver = operations_research::MPSolver::CreateSolver("scip"); | ||
if (solver == nullptr) { | ||
LOG(ERROR) << "SCIP solver is not available"; | ||
return EXIT_SUCCESS; | ||
} else { | ||
return RUN_ALL_TESTS(); | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
MPVariable::integer
to check it's an integer variable on the indicator variable, and possibly test it's bounds (0 & 1).