A push/relabel type max cardinality matching implementation.
(slightly incompatible with bipartite_matching.h)
3 * This file is a part of LEMON, a generic C++ optimization library
5 * Copyright (C) 2003-2006
6 * Egervary Jeno Kombinatorikus Optimalizalasi Kutatocsoport
7 * (Egervary Research Group on Combinatorial Optimization, EGRES).
9 * Permission to use, modify and distribute this software is granted
10 * provided that this copyright notice appears in all copies. For
11 * precise terms see the accompanying LICENSE file.
13 * This software is provided "AS IS" with no warranty of any kind,
14 * express or implied, and with no claim as to its suitability for any
20 #include<lemon/lp_cplex.h>
23 ///\brief Implementation of the LEMON-CPLEX lp solver interface.
26 LpCplex::LpCplex() : LpSolverBase() {
28 // env = CPXopenCPLEXdevelop(&status);
29 env = CPXopenCPLEX(&status);
30 lp = CPXcreateprob(env, &status, "LP problem");
38 LpSolverBase &LpCplex::_newLp()
40 //The first approach opens a new environment
41 LpCplex* newlp=new LpCplex();
45 LpSolverBase &LpCplex::_copyLp() {
46 ///\bug FixID data is not copied!
47 //The first approach opens a new environment
48 LpCplex* newlp=new LpCplex();
49 //The routine CPXcloneprob can be used to create a new CPLEX problem
50 //object and copy all the problem data from an existing problem
51 //object to it. Solution and starting information is not copied.
52 newlp->lp = CPXcloneprob(env, lp, &status);
56 int LpCplex::_addCol()
58 int i = CPXgetnumcols(env, lp);
60 lb[0]=-INF;//-CPX_INFBOUND;
61 ub[0]=INF;//CPX_INFBOUND;
62 status = CPXnewcols(env, lp, 1, NULL, lb, ub, NULL, NULL);
67 int LpCplex::_addRow()
69 //We want a row that is not constrained
71 sense[0]='L';//<= constraint
74 int i = CPXgetnumrows(env, lp);
75 status = CPXnewrows(env, lp, 1, rhs, sense, NULL, NULL);
80 void LpCplex::_eraseCol(int i) {
81 CPXdelcols(env, lp, i, i);
84 void LpCplex::_eraseRow(int i) {
85 CPXdelrows(env, lp, i, i);
88 void LpCplex::_getColName(int col, std::string &name)
92 CPXgetcolname(env, lp, 0, 0, 0, &storespace, col, col);
98 ///\bug return code unchecked for error
99 CPXgetcolname(env, lp, names, buf, storespace, &dontcare, col, col);
103 void LpCplex::_setColName(int col, const std::string &name)
107 names[0] = const_cast<char*>(name.c_str());
108 ///\bug return code unchecked for error
109 CPXchgcolname(env, lp, 1, &col, names);
112 ///\warning Data at index 0 is ignored in the arrays.
113 void LpCplex::_setRowCoeffs(int i, LpRowIterator b, LpRowIterator e)
115 std::vector<int> indices;
116 std::vector<int> rowlist;
117 std::vector<Value> values;
119 for(LpRowIterator it=b; it!=e; ++it) {
120 indices.push_back(it->first);
121 values.push_back(it->second);
122 rowlist.push_back(i);
125 status = CPXchgcoeflist(env, lp, values.size(),
126 &rowlist[0], &indices[0], &values[0]);
129 void LpCplex::_setColCoeffs(int i, LpColIterator b, LpColIterator e)
131 std::vector<int> indices;
132 std::vector<int> collist;
133 std::vector<Value> values;
135 for(LpColIterator it=b; it!=e; ++it) {
136 indices.push_back(it->first);
137 values.push_back(it->second);
138 collist.push_back(i);
141 status = CPXchgcoeflist(env, lp, values.size(),
142 &indices[0], &collist[0], &values[0]);
145 void LpCplex::_setCoeff(int row, int col, Value value)
147 CPXchgcoef(env, lp, row, col, value);
150 void LpCplex::_setColLowerBound(int i, Value value)
158 status = CPXchgbds(env, lp, 1, indices, lu, bd);
162 void LpCplex::_setColUpperBound(int i, Value value)
170 status = CPXchgbds(env, lp, 1, indices, lu, bd);
173 //This will be easier to implement
174 void LpCplex::_setRowBounds(int i, Value lb, Value ub)
177 if (lb==INF || ub==-INF) {
188 CPXchgsense(env, lp, cnt, indices, sense);
189 CPXchgcoef(env, lp, i, -1, ub);
195 CPXchgsense(env, lp, cnt, indices, sense);
196 CPXchgcoef(env, lp, i, -1, lb);
201 CPXchgsense(env, lp, cnt, indices, sense);
202 CPXchgcoef(env, lp, i, -1, lb);
206 CPXchgsense(env, lp, cnt, indices, sense);
207 CPXchgcoef(env, lp, i, -1, lb);
208 CPXchgcoef(env, lp, i, -2, ub-lb);
214 // void LpCplex::_setRowLowerBound(int i, Value value)
216 // //Not implemented, obsolete
219 // void LpCplex::_setRowUpperBound(int i, Value value)
221 // //Not implemented, obsolete
222 // // //TODO Ezt kell meg megirni
223 // // //type of the problem
225 // // status = CPXgetsense(env, lp, sense, i, i);
227 // // status = CPXgetrhs(env, lp, rhs, i, i);
229 // // switch (sense[0]) {
230 // // case 'L'://<= constraint
232 // // case 'E'://= constraint
234 // // case 'G'://>= constraint
236 // // case 'R'://ranged constraint
242 // // status = CPXchgcoef(env, lp, i, -2, value_rng);
245 void LpCplex::_setObjCoeff(int i, Value obj_coef)
247 CPXchgcoef(env, lp, -1, i, obj_coef);
250 void LpCplex::_clearObj()
252 for (int i=0;i< CPXgetnumcols(env, lp);++i){
253 CPXchgcoef(env, lp, -1, i, 0);
257 // The routine returns zero unless an error occurred during the
258 // optimization. Examples of errors include exhausting available
259 // memory (CPXERR_NO_MEMORY) or encountering invalid data in the
260 // CPLEX problem object (CPXERR_NO_PROBLEM). Exceeding a
261 // user-specified CPLEX limit, or proving the model infeasible or
262 // unbounded, are not considered errors. Note that a zero return
263 // value does not necessarily mean that a solution exists. Use query
264 // routines CPXsolninfo, CPXgetstat, and CPXsolution to obtain
265 // further information about the status of the optimization.
266 LpCplex::SolveExitStatus LpCplex::_solve()
269 status = CPXlpopt(env, lp);
270 //status = CPXprimopt(env, lp);
271 #if CPX_VERSION >= 800
278 switch (CPXgetstat(env, lp))
280 case CPX_STAT_OPTIMAL:
281 case CPX_STAT_INFEASIBLE:
282 case CPX_STAT_UNBOUNDED:
290 //We want to exclude some cases
291 switch (CPXgetstat(env, lp)){
293 case CPX_IT_LIM_FEAS:
294 case CPX_IT_LIM_INFEAS:
295 case CPX_TIME_LIM_FEAS:
296 case CPX_TIME_LIM_INFEAS:
308 LpCplex::Value LpCplex::_getPrimal(int i)
311 CPXgetx(env, lp, &x, i, i);
315 LpCplex::Value LpCplex::_getDual(int i)
318 CPXgetpi(env, lp, &y, i, i);
322 LpCplex::Value LpCplex::_getPrimalValue()
325 //method = CPXgetmethod (env, lp);
326 //printf("CPXgetprobtype %d \n",CPXgetprobtype(env,lp));
327 status = CPXgetobjval(env, lp, &objval);
328 //printf("Objective value: %g \n",objval);
331 bool LpCplex::_isBasicCol(int i) {
332 int cstat[CPXgetnumcols(env, lp)];
333 CPXgetbase(env, lp, cstat, NULL);
334 return (cstat[i]==CPX_BASIC);
337 //7.5-os cplex statusai (Vigyazat: a 9.0-asei masok!)
338 // This table lists the statuses, returned by the CPXgetstat() routine, for solutions to LP problems or mixed integer problems. If no solution exists, the return value is zero.
340 // For Simplex, Barrier
342 // Optimal solution found
344 // Problem infeasible
348 // Objective limit exceeded in Phase II
350 // Iteration limit exceeded in Phase II
351 // 6 CPX_IT_LIM_INFEAS
352 // Iteration limit exceeded in Phase I
353 // 7 CPX_TIME_LIM_FEAS
354 // Time limit exceeded in Phase II
355 // 8 CPX_TIME_LIM_INFEAS
356 // Time limit exceeded in Phase I
357 // 9 CPX_NUM_BEST_FEAS
358 // Problem non-optimal, singularities in Phase II
359 // 10 CPX_NUM_BEST_INFEAS
360 // Problem non-optimal, singularities in Phase I
361 // 11 CPX_OPTIMAL_INFEAS
362 // Optimal solution found, unscaled infeasibilities
364 // Aborted in Phase II
365 // 13 CPX_ABORT_INFEAS
366 // Aborted in Phase I
367 // 14 CPX_ABORT_DUAL_INFEAS
368 // Aborted in barrier, dual infeasible
369 // 15 CPX_ABORT_PRIM_INFEAS
370 // Aborted in barrier, primal infeasible
371 // 16 CPX_ABORT_PRIM_DUAL_INFEAS
372 // Aborted in barrier, primal and dual infeasible
373 // 17 CPX_ABORT_PRIM_DUAL_FEAS
374 // Aborted in barrier, primal and dual feasible
375 // 18 CPX_ABORT_CROSSOVER
376 // Aborted in crossover
378 // Infeasible or unbounded
382 // Ezeket hova tegyem:
383 // ??case CPX_ABORT_DUAL_INFEAS
384 // ??case CPX_ABORT_CROSSOVER
385 // ??case CPX_INForUNBD
388 //Some more interesting stuff:
390 // CPX_PARAM_LPMETHOD 1062 int LPMETHOD
395 // 4 Standard Barrier
397 // Description: Method for linear optimization.
398 // Determines which algorithm is used when CPXlpopt() (or "optimize" in the Interactive Optimizer) is called. Currently the behavior of the "Automatic" setting is that CPLEX simply invokes the dual simplex method, but this capability may be expanded in the future so that CPLEX chooses the method based on problem characteristics
400 void statusSwitch(CPXENVptr env,int& stat){
401 #if CPX_VERSION < 900
403 CPXgetintparam (env,CPX_PARAM_LPMETHOD,&lpmethod);
405 if (stat==CPX_UNBOUNDED){
409 if (stat==CPX_INFEASIBLE)
416 LpCplex::SolutionStatus LpCplex::_getPrimalStatus()
418 //Unboundedness not treated well: the following is from cplex 9.0 doc
419 // About Unboundedness
421 // The treatment of models that are unbounded involves a few
422 // subtleties. Specifically, a declaration of unboundedness means that
423 // ILOG CPLEX has determined that the model has an unbounded
424 // ray. Given any feasible solution x with objective z, a multiple of
425 // the unbounded ray can be added to x to give a feasible solution
426 // with objective z-1 (or z+1 for maximization models). Thus, if a
427 // feasible solution exists, then the optimal objective is
428 // unbounded. Note that ILOG CPLEX has not necessarily concluded that
429 // a feasible solution exists. Users can call the routine CPXsolninfo
430 // to determine whether ILOG CPLEX has also concluded that the model
431 // has a feasible solution.
433 int stat = CPXgetstat(env, lp);
434 #if CPX_VERSION >= 800
437 case CPX_STAT_OPTIMAL:
439 case CPX_STAT_UNBOUNDED:
441 case CPX_STAT_INFEASIBLE:
447 statusSwitch(env,stat);
448 //CPXgetstat(env, lp);
449 //printf("A primal status: %d, CPX_OPTIMAL=%d \n",stat,CPX_OPTIMAL);
452 return UNDEFINED; //Undefined
453 case CPX_OPTIMAL://Optimal
455 case CPX_UNBOUNDED://Unbounded
456 return INFEASIBLE;//In case of dual simplex
458 case CPX_INFEASIBLE://Infeasible
459 // case CPX_IT_LIM_INFEAS:
460 // case CPX_TIME_LIM_INFEAS:
461 // case CPX_NUM_BEST_INFEAS:
462 // case CPX_OPTIMAL_INFEAS:
463 // case CPX_ABORT_INFEAS:
464 // case CPX_ABORT_PRIM_INFEAS:
465 // case CPX_ABORT_PRIM_DUAL_INFEAS:
466 return INFINITE;//In case of dual simplex
469 // case CPX_IT_LIM_FEAS:
470 // case CPX_TIME_LIM_FEAS:
471 // case CPX_NUM_BEST_FEAS:
472 // case CPX_ABORT_FEAS:
473 // case CPX_ABORT_PRIM_DUAL_FEAS:
476 return UNDEFINED; //Everything else comes here
482 //9.0-as cplex verzio statusai
483 // CPX_STAT_ABORT_DUAL_OBJ_LIM
484 // CPX_STAT_ABORT_IT_LIM
485 // CPX_STAT_ABORT_OBJ_LIM
486 // CPX_STAT_ABORT_PRIM_OBJ_LIM
487 // CPX_STAT_ABORT_TIME_LIM
488 // CPX_STAT_ABORT_USER
489 // CPX_STAT_FEASIBLE_RELAXED
490 // CPX_STAT_INFEASIBLE
491 // CPX_STAT_INForUNBD
494 // CPX_STAT_OPTIMAL_FACE_UNBOUNDED
495 // CPX_STAT_OPTIMAL_INFEAS
496 // CPX_STAT_OPTIMAL_RELAXED
497 // CPX_STAT_UNBOUNDED
499 LpCplex::SolutionStatus LpCplex::_getDualStatus()
501 int stat = CPXgetstat(env, lp);
502 #if CPX_VERSION >= 800
505 case CPX_STAT_OPTIMAL:
507 case CPX_STAT_UNBOUNDED:
513 statusSwitch(env,stat);
516 return UNDEFINED; //Undefined
517 case CPX_OPTIMAL://Optimal
522 return UNDEFINED; //Everything else comes here
528 LpCplex::ProblemTypes LpCplex::_getProblemType()
530 int stat = CPXgetstat(env, lp);
531 #if CPX_VERSION >= 800
534 case CPX_STAT_OPTIMAL:
535 return PRIMAL_DUAL_FEASIBLE;
536 case CPX_STAT_UNBOUNDED:
537 return PRIMAL_FEASIBLE_DUAL_INFEASIBLE;
543 case CPX_OPTIMAL://Optimal
544 return PRIMAL_DUAL_FEASIBLE;
546 return PRIMAL_FEASIBLE_DUAL_INFEASIBLE;
547 // return PRIMAL_INFEASIBLE_DUAL_FEASIBLE;
548 // return PRIMAL_DUAL_INFEASIBLE;
550 //Seems to be that this is all we can say for sure
559 void LpCplex::_setMax()
561 CPXchgobjsen(env, lp, CPX_MAX);
563 void LpCplex::_setMin()
565 CPXchgobjsen(env, lp, CPX_MIN);