Modeling Adjustable Robust Problems¶
Warning
This tutorial is still under construction.
In this tutorial, we will see how to model a two-stage robust problem in idol.
To follow this tutorial, you should be familiar with two-stage robust optimization and modeling optimization problems in idol. If this is not the case, we recommend you to read the tutorial on MIP modeling.
A Two-Stage Robust Facility Location Problem¶
We consider a two-stage robust facility location problem (FLP) where demands are uncertain.
Given a set of potential facility locations
Note that there is also an example for the deterministic version of the FLP using Column Generation.
Each facility
In this robust variant, we consider that the demands are uncertain and can be expressed as
We model the two-stage robust FLP as
where
In what follows, we will assume that we have a variable instance
of type idol::Problems::FLP::Instance
that contains the data of the problem. Most typically, you will want to read the instance from a file. This is done as follows.
// Read instance
const auto instance = Problems::FLP::read_instance_1991_Cornuejols_eal("robust_ccg.data.txt");
Additionally, we define an optimization environment env
and some parameters.
Env env;
const double Gamma = 3;
const double percentage_increase = .2;
Modeling Steps¶
To model a two-stage robust problem, one needs to perform the following steps:
Define an uncertainty set
.Define the deterministic model in which
is a parameter.Assign a stage to each variable and constraint.
Let’s see how these steps are done in idol.
Defining the Uncertainty Set¶
Modeling the uncertainty set
const unsigned int n_customers = instance.n_customers();
Model uncertainty_set(env);
auto xi = uncertainty_set.add_vars(Dim<1>(n_customers), 0., 1, Binary, "xi");
uncertainty_set.add_ctr(idol_Sum(j, Range(n_customers), xi[j]) <= Gamma);
Note that defining an objective function is not necessary since the uncertainty set is not optimized over. If an objective function is defined, it will be ignored by the optimizer.
Defining the Deterministic Model¶
The deterministic model underlying the two-stage robust FLP is the same as the classical FLP, except that the demand is seen as a parameter.
Recall that a variable, e.g., xi[0]
, can be turned into a parameter by prepending it with !
.
Hence, we can define the deterministic model as follows.
const unsigned int n_facilities = instance.n_facilities();
Model model(env);
const auto x = model.add_vars(Dim<1>(n_facilities), 0., 1., Binary, "x");
const auto y = model.add_vars(Dim<2>(n_facilities, n_customers), 0., Inf, Continuous, "y");
// Capacity constraints
for (unsigned int i = 0 ; i < n_facilities ; ++i) {
model.add_ctr(idol_Sum(j, Range(n_customers), y[i][j]) <= instance.capacity(i) * x[i]);
}
// Demand satisfaction constraints
for (unsigned int j = 0 ; j < n_customers ; ++j) {
// IMPORTANT: here we use the parameter "!xi[j]" instead of the variable "xi[j]"
model.add_ctr(idol_Sum(i, Range(n_facilities), y[i][j]) == instance.demand(j) * (1 + percentage_increase * !xi[j]));
}
// Objective function
model.seobj_expr(idol_Sum(i, Range(n_facilities),
instance.fixed_cost(i) * x[i]
+ idol_Sum(j, Range(n_customers),
instance.per_unit_transportation_cost(i, j) * y[i][j]
)
)
);
Assigning Stages¶
The last step is to assign a stage to each variable and constraint. Here, variables
Assigning stages is done by creating a new object of type idol::Robust::StageDescription
. Under the hood, this object does nothing more but defining new annotations for variables and constraints storing the assigned stage of each variable and constraint. It is created as follows.
Robust::StageDescription stages(env);
By default, all variables and constraints are assigned to the first stage. To assign a variable or constraint to the second stage, one can use the method set_stage
of the object stages
. For instance, one can do as follows.
for (const auto& var : model.vars()) {
if (var.name().front() != 'x') {
stages.set_stage(var, 2);
}
}
Similarly, since all constraints are second-stage constraints, one can do as follows.
for (const auto& ctr : model.ctrs()) {
stages.set_stage(ctr, 2);
}
About stage annotations
Note that it is also possible to define your own annotations to assign variables and constraints to stages. This is a rather advanced feature and it is your responsability to ensure that the annotations are consistent with the model.
The annotations are based on the following conventions: all first-stage variables and constraints have the annotation evaluating to MasterId
. All second-stage variables and constraints have the annotation evaluating to 0
.
For instance, the following code is equivalent to the previous one.
Annotation<Var, unsigned int> stage_vars(model, "stage_vars", MasterId); // By default, all variables are first-stage variables
Annotation<Ctr, unsigned int> stage_ctrs(model, "stage_ctrs", MasterId); // By default, all constraints are first-stage constraints
for (const auto& var : model.vars()) {
if (var.name().front() != 'x') {
var.set(stage_vars, 0); // Assign variable to the second stage
}
}
for (const auto& ctr : model.ctrs()) {
ctr.set(stage_ctrs, 0); // Assign constraint to the second stage
}
idol::Robust::StageDescription stages(stage_vars, stage_ctrs);
By doing so, a call to stages.stage(var)
will return “1” for all first-stage variables and “2” for all second-stage variables. The underlying annotation can be obtained using
Annotation<Var, unsigned int> stage_vars = stages.stage_vars()
Finally, also note the method stages.stage_index(var)
that will return “0” for all first-stage variables and “1” for all second-stage variables.
That’s it! We have now modeled a two-stage robust FLP in idol. Note that you will now need to attach an optimizer to the model to solve it. To this end, be sure to check the tutorials on optimizers for two-stage robust problems, e.g., the column-and-constraint generation tutorial.
Complete Example¶
A complete example is given here