5.5.6. MIQP Modeling and Optimization in C#

In this chapter, we will use MindOpt C# API to model and solve the problem in Example of Mixed Integer Quadratic Programming.

Create an optimization model model:

32            // Create model
33            MDOEnv env = new MDOEnv(); 
34            MDOModel model = new MDOModel(env); 
35            model.Set(MDO.StringAttr.ModelName, "MIQP_01");

Next, we set the optimization sense to minimization via MDOModel.Set . Then, we call MDOModel.AddVar to add four variables, which define upper bounds, lower bounds, names and types. (for more details on MDOModel.Set and MDOModel.AddVar, please refer to C# API)

42                // Add variables.
43                MDOVar[] x = new MDOVar[4];
44                x[0] = model.AddVar(0.0,         10.0, 1.0, 'I', "x0");
45                x[1] = model.AddVar(0.0, MDO.INFINITY, 1.0, 'C', "x1");
46                x[2] = model.AddVar(0.0, MDO.INFINITY, 1.0, 'C', "x2");
47                x[3] = model.AddVar(0.0, MDO.INFINITY, 1.0, 'C', "x3");

We input the linear constraints into model:

49                // Add constraints.
50                double[][] consV = new double[][]
51                {
52                    new double[] { 1.0, 1.0, 2.0, 3.0},
53                    new double[] { 1.0, 0,  -1.0, 6.0} 
54                };
55
56                MDOLinExpr tempLinExpr1 = new MDOLinExpr();
57                tempLinExpr1.AddTerms(consV[0], x);
58                model.AddConstr(tempLinExpr1, MDO.GREATER_EQUAL, 1.0, "c0");
59
60                MDOLinExpr tempLinExpr2 = new MDOLinExpr();
61                tempLinExpr2.AddTerms(consV[1], x);
62                model.AddConstr(tempLinExpr2, MDO.EQUAL, 1.0, "c1");   

Then, we create a quadratic expression MDOQuadExpr and call MDOQuadExpr.AddTerms to set the linear part of the objective function. Here obj_idx represents the indices of the linear terms, obj_val represents the corresponding non-zero coefficient values in obj_idx.

64                // Create a QuadExpr for quadratic objective
65                MDOQuadExpr obj = new MDOQuadExpr();
66
67                // Add objective linear term: 1 x0 + 1 x1 + 1 x2 + 1 x3 
68                int      obj_nnz = 4;
69                MDOVar[] obj_idx = new MDOVar[] { x[0], x[1], x[2], x[3] };
70                double[] obj_val = new double[] { 1.0,  1.0,  1.0,  1.0 };
71                obj.AddTerms(obj_val, obj_idx);

We call MDOQuadExpr.AddTerms to set the quadratic terms of the objective. Here, qo_values represents the coefficients of all the non-zero quadratic terms, while qo_col1 and qo_col2 respectively represent its row and column indices.

73                // Add quadratic part in objective: 1/2 [ x0^2 + x1^2 + x2^2 + x3^2 + x0 x1] 
74                int      qo_nnz    = 5;
75                MDOVar[] qo_col1   = new MDOVar[] { x[0], x[1], x[2], x[3], x[0] };
76                MDOVar[] qo_col2   = new MDOVar[] { x[0], x[1], x[2], x[3], x[1] };
77                double[] qo_values = new double[] { 0.5,  0.5,  0.5,  0.5,  0.5 };
78                obj.AddTerms(qo_values, qo_col1, qo_col2);

Lastly, we call MDOModel.SetObjective to set the objective and the direction to be optimized.

80                model.SetObjective(obj, MDO.MINIMIZE);

Once the model is constructed, we call MDOModel.Optimize to solve the problem:

83                model.Optimize();

The complete example code is shown in MdoMIQPEx1.cs :

  1/**
  2 *  Description
  3 *  -----------
  4 *  Linear optimization (row-wise input).
  5 *
  6 *  Formulation
  7 *  -----------
  8 *
  9 *  Minimize
 10 *    obj: 1 x0 + 1 x1 + 1 x2 + 1 x3 
 11 *         + 1/2 [ x0^2 + x1^2 + x2^2 + x3^2 + x0 x1]
 12 *  Subject To
 13 *   c0 : 1 x0 + 1 x1 + 2 x2 + 3 x3 >= 1
 14 *   c1 : 1 x0 - 1 x2 + 6 x3 = 1
 15 *  Bounds
 16 *    0 <= x0 <= 10
 17 *    0 <= x1
 18 *    0 <= x2
 19 *    0 <= x3
 20  *  x0 Integer
 21 *  End
 22 */
 23
 24using Mindopt;
 25
 26namespace Example
 27{
 28    public class MdoQoEx1
 29    {
 30        public static void Main(string[] args)
 31        {
 32            // Create model
 33            MDOEnv env = new MDOEnv(); 
 34            MDOModel model = new MDOModel(env); 
 35            model.Set(MDO.StringAttr.ModelName, "MIQP_01");
 36
 37            try
 38            {
 39                // Change to minimization problem.
 40                model.Set(MDO.IntAttr.ModelSense, MDO.MINIMIZE);
 41
 42                // Add variables.
 43                MDOVar[] x = new MDOVar[4];
 44                x[0] = model.AddVar(0.0,         10.0, 1.0, 'I', "x0");
 45                x[1] = model.AddVar(0.0, MDO.INFINITY, 1.0, 'C', "x1");
 46                x[2] = model.AddVar(0.0, MDO.INFINITY, 1.0, 'C', "x2");
 47                x[3] = model.AddVar(0.0, MDO.INFINITY, 1.0, 'C', "x3");
 48
 49                // Add constraints.
 50                double[][] consV = new double[][]
 51                {
 52                    new double[] { 1.0, 1.0, 2.0, 3.0},
 53                    new double[] { 1.0, 0,  -1.0, 6.0} 
 54                };
 55
 56                MDOLinExpr tempLinExpr1 = new MDOLinExpr();
 57                tempLinExpr1.AddTerms(consV[0], x);
 58                model.AddConstr(tempLinExpr1, MDO.GREATER_EQUAL, 1.0, "c0");
 59
 60                MDOLinExpr tempLinExpr2 = new MDOLinExpr();
 61                tempLinExpr2.AddTerms(consV[1], x);
 62                model.AddConstr(tempLinExpr2, MDO.EQUAL, 1.0, "c1");   
 63
 64                // Create a QuadExpr for quadratic objective
 65                MDOQuadExpr obj = new MDOQuadExpr();
 66
 67                // Add objective linear term: 1 x0 + 1 x1 + 1 x2 + 1 x3 
 68                int      obj_nnz = 4;
 69                MDOVar[] obj_idx = new MDOVar[] { x[0], x[1], x[2], x[3] };
 70                double[] obj_val = new double[] { 1.0,  1.0,  1.0,  1.0 };
 71                obj.AddTerms(obj_val, obj_idx);
 72
 73                // Add quadratic part in objective: 1/2 [ x0^2 + x1^2 + x2^2 + x3^2 + x0 x1] 
 74                int      qo_nnz    = 5;
 75                MDOVar[] qo_col1   = new MDOVar[] { x[0], x[1], x[2], x[3], x[0] };
 76                MDOVar[] qo_col2   = new MDOVar[] { x[0], x[1], x[2], x[3], x[1] };
 77                double[] qo_values = new double[] { 0.5,  0.5,  0.5,  0.5,  0.5 };
 78                obj.AddTerms(qo_values, qo_col1, qo_col2);
 79
 80                model.SetObjective(obj, MDO.MINIMIZE);
 81                
 82                // Solve the problem and populate optimization result.
 83                model.Optimize();
 84
 85                if (model.Get(MDO.IntAttr.Status) == MDO.Status.OPTIMAL)
 86                {
 87                    Console.WriteLine($"Optimal objective value is: {model.Get(MDO.DoubleAttr.ObjVal)}");
 88                    Console.WriteLine("Decision variables: ");
 89                    for (int i = 0; i < 4; i++)
 90                        Console.WriteLine( $"x[{i}] = {x[i].Get(MDO.DoubleAttr.X)}");
 91                }
 92                else
 93                {
 94                    Console.WriteLine("No feasible solution.");
 95                }
 96            }
 97            catch (Exception e)
 98            { 
 99                Console.WriteLine("Exception during optimization");
100                Console.WriteLine(e.Message);
101            }
102            finally
103            { 
104                model.Dispose();
105                env.Dispose();
106            }
107        }
108    }
109}