SoPlex Doxygen Documentation
spxlp.h
Go to the documentation of this file.
1 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
2 /* */
3 /* This file is part of the class library */
4 /* SoPlex --- the Sequential object-oriented simPlex. */
5 /* */
6 /* Copyright (C) 1996-2012 Konrad-Zuse-Zentrum */
7 /* fuer Informationstechnik Berlin */
8 /* */
9 /* SoPlex is distributed under the terms of the ZIB Academic Licence. */
10 /* */
11 /* You should have received a copy of the ZIB Academic License */
12 /* along with SoPlex; see the file COPYING. If not email to soplex@zib.de. */
13 /* */
14 /* * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * * */
15 
16 /**@file spxlp.h
17  * @brief Saving LPs in a form suitable for SoPlex.
18  */
19 #ifndef _SPXLP_H_
20 #define _SPXLP_H_
21 
22 #include <assert.h>
23 #include <iostream>
24 #include <iomanip>
25 
26 #include "spxdefines.h"
27 #include "datakey.h"
28 #include "spxid.h"
29 #include "dvector.h"
30 #include "dvector_exact.h"
31 #include "svset.h"
32 #include "dataarray.h"
33 #include "lprow.h"
34 #include "lpcol.h"
35 #include "lprowset.h"
36 #include "lpcolset.h"
37 #include "nameset.h"
38 
39 namespace soplex
40 {
41 class SPxSolver;
42 
43 /**@brief Saving LPs in a form suitable for SoPlex.
44  @ingroup Algo
45 
46  Class SPxLP provides the data structures required for saving a
47  linear program in the form
48  \f[
49  \begin{array}{rl}
50  \hbox{max} & c^T x \\
51  \hbox{s.t.} & l_r \le Ax \le u_r \\
52  & l_c \le x \le u_c
53  \end{array}
54  \f]
55  suitable for solving with SoPlex. This includes:
56  - SVSets for both, columns and rows
57  - objective Vector
58  - upper and lower bound Vectors for variables (\f$l_c\f$ and \f$u_c\f$)
59  - upper and lower bound Vectors for inequalities (\f$l_r\f$ and \f$u_r\f$)
60 
61  Note, that the optimization sense is not saved directly. Instead, the
62  objective function are multiplied by -1 to transform the LP to our standard
63  form maximizing the objective function. However, the sense of the loaded LP
64  can be retreived with method #spxSense().
65 
66  Further, equality constraints are modelled by \f$l_r = u_r\f$.
67  Analogously, fixed variables have \f$l_c = u_c\f$.
68 
69  #SPxLP%s are saved as an SVSet, both for the columns and rows.
70  Note that this is redundant but eases the access.
71 */
72 class SPxLP : protected LPRowSet, protected LPColSet
73 {
74  friend class SPxBasis;
75  friend class SPxScaler;
76  friend class SPxEquiliSC;
77  friend class SPxGeometSC;
78  friend class SPxMainSM;
79 
80 public:
81 
82  //---------------------------
83  /**@name Types */
84  //@{
85  /// optimization sense.
86  enum SPxSense
87  {
88  MAXIMIZE = 1,
89  MINIMIZE = -1
90  };
91  //@}
92 
93 private:
94 
95  //---------------------------
96  /**@name Data */
97  //@{
98  SPxSense thesense; ///< optimization sense.
99  //@}
100 
101 public:
102 
103  //---------------------------
104  /**@name Inquiry */
105  //@{
106  /// returns number of rows in LP.
107  int nRows() const
108  {
109  return LPRowSet::num();
110  }
111 
112  /// returns number of columns in LP.
113  int nCols() const
114  {
115  return LPColSet::num();
116  }
117 
118  /// number of nonzeros in LP.
119  int nNzos() const;
120 
121  /// absolute smallest non-zero element in LP.
122  Real minAbsNzo() const;
123  /// absolute biggest non-zero element in LP.
124  Real maxAbsNzo() const;
125 
126  /// gets \p i 'th row.
127  void getRow(int i, LPRow& row) const;
128 
129  /// gets row with identifier \p id.
130  void getRow(const SPxRowId& id, LPRow& row) const
131  {
132  getRow(number(id), row);
133  }
134 
135  /// gets rows \p start, ... \p end.
136  void getRows(int start, int end, LPRowSet& set) const;
137 
138  /// gets row vector of row \p i.
139  const SVector& rowVector(int i) const
140  {
141  return LPRowSet::rowVector(i);
142  }
143 
144  /// gets row vector of row with identifier \p id.
145  const SVector& rowVector(const SPxRowId& id) const
146  {
147  return LPRowSet::rowVector(id);
148  }
149 
150  /// returns right hand side vector.
151  const Vector& rhs() const
152  {
153  return LPRowSet::rhs();
154  }
155 
156  ///
157  Real rhs(int i) const
158  {
159  return LPRowSet::rhs(i);
160  }
161 
162  /// returns right hand side of row with identifier \p id.
163  Real rhs(const SPxRowId& id) const
164  {
165  return LPRowSet::rhs(id);
166  }
167 
168  /// returns left hand side vector.
169  const Vector& lhs() const
170  {
171  return LPRowSet::lhs();
172  }
173 
174  ///
175  Real lhs(int i) const
176  {
177  return LPRowSet::lhs(i);
178  }
179 
180  /// returns left hand side of row with identifier \p id.
181  Real lhs(const SPxRowId& id) const
182  {
183  return LPRowSet::lhs(id);
184  }
185 
186  /// returns the inequality type of the \p i'th LPRow.
187  LPRow::Type rowType(int i) const
188  {
189  return LPRowSet::type(i);
190  }
191 
192  /// returns the inequality type of the row with identifier \p key.
193  LPRow::Type rowType(const SPxRowId& id) const
194  {
195  return LPRowSet::type(id);
196  }
197 
198  /// gets \p i 'th column.
199  void getCol(int i, LPCol& column) const;
200 
201  /// gets column with identifier \p id.
202  void getCol(const SPxColId& id, LPCol& col) const
203  {
204  getCol(number(id), col);
205  }
206 
207  /// gets columns \p start, ..., \p end.
208  void getCols(int start, int end, LPColSet& set) const;
209 
210  /// returns column vector of column \p i.
211  const SVector& colVector(int i) const
212  {
213  return LPColSet::colVector(i);
214  }
215 
216  /// returns column vector of column with identifier \p id.
217  const SVector& colVector(const SPxColId& id) const
218  {
219  return LPColSet::colVector(id);
220  }
221 
222  /// gets objective vector.
223  void getObj(Vector& obj) const;
224 
225  /// returns objective value of column \p i.
226  Real obj(int i) const
227  {
228  return Real(spxSense()) * maxObj(i);
229  }
230 
231  /// returns objective value of column with identifier \p id.
232  Real obj(const SPxColId& id) const
233  {
234  return Real(spxSense()) * maxObj(id);
235  }
236 
237  /// returns objective vector for maximization problem.
238  /** Methods #maxObj() return the objective vector or its elements, after
239  transformation to a maximization problem. Since this is how SPxLP
240  internally stores any LP these methods are generally faster. The
241  following condition holds: #obj() = #spxSense() * maxObj().
242  */
243  const Vector& maxObj() const
244  {
245  return LPColSet::maxObj();
246  }
247 
248  /// returns objective value of column \p i for maximization problem.
249  Real maxObj(int i) const
250  {
251  return LPColSet::maxObj(i);
252  }
253 
254  /// returns objective value of column with identifier \p id
255  /// for maximization problem.
256  Real maxObj(const SPxColId& id) const
257  {
258  return LPColSet::maxObj(id);
259  }
260 
261  /// returns upper bound vector.
262  const Vector& upper() const
263  {
264  return LPColSet::upper();
265  }
266 
267  /// returns upper bound of column \p i.
268  Real upper(int i) const
269  {
270  return LPColSet::upper(i);
271  }
272  /// returns upper bound of column with identifier \p id.
273  Real upper(const SPxColId& id) const
274  {
275  return LPColSet::upper(id);
276  }
277 
278  /// returns lower bound vector.
279  const Vector& lower() const
280  {
281  return LPColSet::lower();
282  }
283 
284  /// returns lower bound of column \p i.
285  Real lower(int i) const
286  {
287  return LPColSet::lower(i);
288  }
289 
290  /// returns lower bound of column with identifier \p id.
291  Real lower(const SPxColId& id) const
292  {
293  return LPColSet::lower(id);
294  }
295 
296  /// returns the optimization sense.
298  {
299  return thesense;
300  }
301 
302  /// returns the row number of the row with identifier \p id.
303  int number(const SPxRowId& id) const
304  {
305  return LPRowSet::number(id);
306  }
307 
308  /// returns the column number of the column with identifier \p id.
309  int number(const SPxColId& id) const
310  {
311  return LPColSet::number(id);
312  }
313 
314  /// returns the row or column number for identifier \p id.
315  int number(const SPxId& id) const
316  {
317  return (id.type() == SPxId::COL_ID)
318  ? LPColSet::number(id)
319  : LPRowSet::number(id);
320  }
321 
322  /// returns the row identifier for row \p n.
323  SPxRowId rId(int n) const
324  {
325  return SPxRowId(LPRowSet::key(n));
326  }
327 
328  /// returns the column identifier for column \p n.
329  SPxColId cId(int n) const
330  {
331  return SPxColId(LPColSet::key(n));
332  }
333  //@}
334 
335 
336  //--------------------------
337  /**@name Extension */
338  //@{
339  ///
340  virtual void addRow(const LPRow& row)
341  {
342  doAddRow(row);
343  }
344  /// adds \p row to LPRowSet.
345  virtual void addRow(SPxRowId& id, const LPRow& row)
346  {
347  addRow(row);
348  id = rId(nRows() - 1);
349  }
350 
351  ///
352  virtual void addRows(const LPRowSet& pset)
353  {
354  doAddRows(pset);
355  }
356  /// adds all LPRows of \p pset to LPRowSet.
357  virtual void addRows(SPxRowId id[], const LPRowSet& set);
358 
359  ///
360  virtual void addCol(const LPCol& col)
361  {
362  doAddCol(col);
363  }
364  /// adds \p col to LPColSet.
365  virtual void addCol(SPxColId& id, const LPCol& col)
366  {
367  addCol(col);
368  id = cId(nCols() - 1);
369  }
370 
371  ///
372  virtual void addCols(const LPColSet& pset)
373  {
374  doAddCols(pset);
375  }
376  /// adds all LPCols of \p set to LPColSet.
377  virtual void addCols(SPxColId id[], const LPColSet& set);
378 
379  //@}
380 
381 
382  //--------------------------
383  /**@name Shrinking */
384  //@{
385  /// removes \p i 'th row.
386  virtual void removeRow(int i)
387  {
388  doRemoveRow(i);
389  }
390 
391  /// removes row with identifier \p id.
392  virtual void removeRow(SPxRowId id)
393  {
394  removeRow(number(id));
395  }
396 
397  /// removes multiple rows.
398  /** This method removes all LPRows from the SPxLP with an
399  index \p i such that \p perm[i] < 0. Upon completion, \p perm[i] >= 0
400  indicates the new index where the \p i'th LPRow has been moved to
401  due to this removal. Note that \p perm must point to an array of at
402  least #nRows() ints.
403  */
404  virtual void removeRows(int perm[])
405  {
406  doRemoveRows(perm);
407  }
408 
409  ///
410  virtual void removeRows(SPxRowId id[], int n, int perm[] = 0);
411  /// removes \p n LPRows.
412  /** Removing multiple rows with one method invocation is available in
413  two flavours. An array \p perm can be passed as third argument or
414  not. If given, \p perm must be an array at least of size #nRows(). It
415  is used to return the permutations resulting from this removal:
416  \p perm[i] < 0 indicates, that the element to index \p i has been
417  removed. Otherwise, \p perm[i] is the new index of the element with
418  index \p i before the removal.
419  */
420  virtual void removeRows(int nums[], int n, int perm[] = 0);
421 
422  /// removes rows from \p start to \p end (including both).
423  virtual void removeRowRange(int start, int end, int perm[] = 0);
424 
425  /// removes \p i 'th column.
426  virtual void removeCol(int i)
427  {
428  doRemoveCol(i);
429  }
430 
431  /// removes column with identifier \p id.
432  virtual void removeCol(SPxColId id)
433  {
434  removeCol(number(id));
435  }
436 
437  /// removes multiple columns.
438  /** This method removes all LPCols from the SPxLP with an
439  index \p i such that \p perm[i] < 0. Upon completion, \p perm[i] >= 0
440  indicates the new index where the \p i 'th LPCol has been moved to
441  due to this removal. Note, that \p perm must point to an array of at
442  least #nCols() ints.
443  */
444  virtual void removeCols(int perm[])
445  {
446  doRemoveCols(perm);
447  }
448 
449  ///
450  virtual void removeCols(SPxColId id[], int n, int perm[] = 0);
451  /// removes \p n LPCols.
452  /** Removing multiple columns with one method invocation is available in
453  two flavours. An array \p perm can be passed as third argument or
454  not. If given, \p perm must be an array at least of size #nCols(). It
455  is used to return the permutations resulting from this removal:
456  \p perm[i] < 0 indicates, that the element to index \p i has been
457  removed. Otherwise, \p perm[i] is the new index of the element with
458  index \p i before the removal.
459  */
460  virtual void removeCols(int nums[], int n, int perm[] = 0);
461 
462  /// removes columns from \p start to \p end (including both).
463  virtual void removeColRange(int start, int end, int perm[] = 0);
464 
465  /// clears the LP.
466  virtual void clear();
467  //@}
468 
469 
470  //--------------------------
471  /**@name IO */
472  //@{
473  /// reads a file in LP format from \p in.
474  virtual bool readLPF( std::istream& in,
475  NameSet* rowNames = 0,
476  NameSet* colNames = 0,
477  DIdxSet* intVars = 0 );
478 
479  /// reads a file from input stream \p in.
480  virtual bool read( std::istream& in,
481  NameSet* rowNames = 0,
482  NameSet* colNames = 0,
483  DIdxSet* intVars = 0 );
484 
485  /// reads a file from a file.
486  virtual bool readFile( const char* filename,
487  NameSet* rowNames = 0,
488  NameSet* colNames = 0,
489  DIdxSet* intVars = 0 );
490 
491 
492  /** Writes a file in LP format to \p out. If \p rowNames and \p
493  * colNames are \c NULL, default names are used for the constraints and
494  * variables. If \p intVars is not \c NULL, the variables contained in
495  * it are marked as integer in the output.
496  */
497  virtual void writeLPF( std::ostream& out,
498  const NameSet* rowNames,
499  const NameSet* colNames,
500  const DIdxSet* p_intvars = 0 ) const;
501 
502  /// Write loaded LP to \p filename.
503  virtual void writeFile(const char* filename,
504  const NameSet* rowNames = 0,
505  const NameSet* colNames = 0,
506  const DIdxSet* p_intvars = 0 ) const;
507 
508 
509  /// Reads a file in MPS format from \p in.
510  virtual bool readMPS( std::istream& in,
511  NameSet* rowNames = 0,
512  NameSet* colNames = 0,
513  DIdxSet* intVars = 0 );
514 
515  /// Writes a file in MPS format to \p out.
516  virtual void writeMPS( std::ostream& out,
517  const NameSet* rowNames,
518  const NameSet* colNames,
519  const DIdxSet* p_intvars = 0 ) const;
520  //@}
521 
522 
523  //--------------------------
524  /**@name Manipulation */
525  //@{
526  /// changes objective vector to \p newObj.
527  virtual void changeObj(const Vector& newObj);
528 
529  /// changes \p i 'th objective vector element to \p newVal.
530  virtual void changeObj(int i, Real newVal);
531 
532  /// change objective value of column with identifier \p id to \p newVal.
533  virtual void changeObj(SPxColId id, Real newVal)
534  {
535  changeObj(number(id), newVal);
536  }
537 
538  /// changes vector of lower bounds to \p newLower.
539  virtual void changeLower(const Vector& newLower);
540 
541  /// changes \p i 'th lower bound to \p newLower.
542  virtual void changeLower(int i, Real newLower);
543 
544  /// changes lower bound of column with identifier \p id to \p newLower.
545  virtual void changeLower(SPxColId id, Real newLower)
546  {
547  changeLower(number(id), newLower);
548  }
549 
550  /// changes vector of upper bounds to \p newUpper.
551  virtual void changeUpper(const Vector& newUpper);
552 
553  /// changes \p i 'th upper bound to \p newUpper.
554  virtual void changeUpper(int i, Real newUpper);
555 
556  /// changes upper bound of column with identifier \p id to \p newLower.
557  virtual void changeUpper(SPxColId id, Real newUpper)
558  {
559  changeUpper(number(id), newUpper);
560  }
561 
562  /// changes variable bounds to \p newLower and \p newUpper.
563  virtual void changeBounds(const Vector& newLower, const Vector& newUpper);
564 
565  /// changes bounds of column \p i to \p newLower and \p newUpper.
566  virtual void changeBounds(int i, Real newLower, Real newUpper);
567 
568  /// changes bounds of column with identifier \p id.
569  virtual void changeBounds(SPxColId id, Real newLower, Real newUpper)
570  {
571  changeBounds(number(id), newLower, newUpper);
572  }
573 
574  /// changes left hand side vector for constraints to \p newLhs.
575  virtual void changeLhs(const Vector& newLhs);
576 
577  /// changes \p i 'th left hand side value to \p newLhs.
578  virtual void changeLhs(int i, Real newLhs);
579 
580  /// changes left hand side value for row with identifier \p id.
581  virtual void changeLhs(SPxRowId id, Real newLhs)
582  {
583  changeLhs(number(id), newLhs);
584  }
585 
586  /// changes right hand side vector for constraints to \p newRhs.
587  virtual void changeRhs(const Vector& newRhs);
588 
589  /// changes \p i 'th right hand side value to \p newRhs.
590  virtual void changeRhs(int i, Real newRhs);
591 
592  /// changes right hand side value for row with identifier \p id.
593  virtual void changeRhs(SPxRowId id, Real newRhs)
594  {
595  changeRhs(number(id), newRhs);
596  }
597 
598  /// changes left and right hand side vectors.
599  virtual void changeRange(const Vector& newLhs, const Vector& newRhs);
600 
601  /// changes left and right hand side of row \p i.
602  virtual void changeRange(int i, Real newLhs, Real newRhs);
603 
604  /// changes left and right hand side of row with identifier \p id.
605  virtual void changeRange(SPxRowId id, Real newLhs, Real newRhs)
606  {
607  changeRange(number(id), newLhs, newRhs);
608  }
609 
610  /// replaces \p i 'th row of LP with \p newRow.
611  virtual void changeRow(int i, const LPRow& newRow);
612 
613  /// replaces row with identifier \p id with \p newRow.
614  virtual void changeRow(SPxRowId id, const LPRow& newRow)
615  {
616  changeRow(number(id), newRow);
617  }
618 
619  /// replaces \p i 'th column of LP with \p newCol.
620  virtual void changeCol(int i, const LPCol& newCol);
621 
622  /// replaces column with identifier \p id with \p newCol.
623  virtual void changeCol(SPxColId id, const LPCol& newCol)
624  {
625  changeCol(number(id), newCol);
626  }
627 
628  /// changes LP element (\p i, \p j) to \p val.
629  virtual void changeElement(int i, int j, Real val);
630 
631  /// changes LP element identified by (\p rid, \p cid) to \p val.
632  virtual void changeElement(SPxRowId rid, SPxColId cid, Real val)
633  {
634  changeElement(number(rid), number(cid), val);
635  }
636 
637  /// changes optimization sense to \p sns.
638  virtual void changeSense(SPxSense sns)
639  {
640  if (sns != thesense)
641  LPColSet::maxObj_w() *= -1.0;
642  thesense = sns;
643  }
644 
645  /// compute activity of the rows for a given primal vector exactly.
646  /// @throw SPxInternalCodeException if dimension of primal vector does not match number of columns
647  virtual DVector_exact computePrimalActivity(const Vector_exact& primal) const;
648 
649  /// compute "dual" activity of the columns for a given dual vector, i.e., y^T A, exactly
650  /// @throw SPxInternalCodeException if dimension of dual vector does not match number of rows
651  virtual DVector_exact computeDualActivity(const Vector_exact& dual) const;
652  //@}
653 
654  //--------------------------
655  /**@name Miscellaneous */
656  //@{
657  /// consistency check.
658  bool isConsistent() const;
659  //@}
660 
661 protected:
662 
663  //--------------------------------
664  /**@name Protected write access */
665  //@{
666  /// returns right hand side of row \p i.
667  Real& rhs_w(int i)
668  {
669  return LPRowSet::rhs_w(i);
670  }
671  /// returns left hand side of row \p i.
672  Real& lhs_w(int i)
673  {
674  return LPRowSet::lhs_w(i);
675  }
676  /// returns objective value of column \p i for maximization problem.
677  Real& maxObj_w(int i)
678  {
679  return LPColSet::maxObj_w(i);
680  }
681  /// returns upper bound of column \p i.
682  Real& upper_w(int i)
683  {
684  return LPColSet::upper_w(i);
685  }
686  /// returns lower bound of column \p i.
687  Real& lower_w(int i)
688  {
689  return LPColSet::lower_w(i);
690  }
691  //@}
692 
693  //--------------------------------
694  /**@name Protected helpers */
695  //@{
696  /// returns the LP as an LPRowSet.
697  const LPRowSet* lprowset() const
698  {
699  return static_cast<const LPRowSet*>(this);
700  }
701 
702  /// returns the LP as an LPColSet.
703  const LPColSet* lpcolset() const
704  {
705  return static_cast<const LPColSet*>(this);
706  }
707  /// internal helper method
708  virtual void doRemoveRow(int i);
709  /// internal helper method
710  virtual void doRemoveCols(int perm[]);
711  /// internal helper method
712  virtual void doRemoveRows(int perm[]);
713  /// internal helper method
714  virtual void doRemoveCol(int i);
715 
716  /// called after the last \p n rows have just been added.
717  virtual void addedRows(int)
718  {}
719  /// called after the last \p n columns have just been added.
720  virtual void addedCols(int)
721  {}
722  ///
723  void added2Set(SVSet& set, const SVSet& add, int n);
724  //@}
725 
726 
727 private:
728 
729  //--------------------------------
730  /**@name Private helpers */
731  //@{
732  /// returns the LP as an LPRowSet.
734  {
735  return LPColSet::colVector_w(i);
736  }
737  ///
739  {
740  return LPRowSet::rowVector_w(i);
741  }
742  ///
743  void doAddRow (const LPRow& row);
744  ///
745  void doAddRows(const LPRowSet& set);
746  ///
747  void doAddCol (const LPCol& col);
748  ///
749  void doAddCols(const LPColSet& set);
750  //@}
751 
752 public:
753 
754  //---------------------------------------
755  /**@name Constructors / Destructors */
756  //@{
757  /// default constructor.
759  {
760  SPxLP::clear(); // clear is virtual.
761 
762  assert(isConsistent());
763  }
764 
765  /// destructor.
766  virtual ~SPxLP()
767  {}
768 
769  /// copy constructor.
770  SPxLP(const SPxLP& old)
771  : LPRowSet(old)
772  , LPColSet(old)
773  , thesense(old.thesense)
774  {
775  assert(isConsistent());
776  }
777 
778  /// assignment operator
779  SPxLP& operator=(const SPxLP& old)
780  {
781  if (this != &old)
782  {
783  LPRowSet::operator=(old);
784  LPColSet::operator=(old);
785  thesense = old.thesense;
786 
787  assert(isConsistent());
788  }
789  return *this;
790  }
791  //@}
792 };
793 
794 
795 } // namespace soplex
796 #endif // _SPXLP_H_
797 
798 //-----------------------------------------------------------------------------
799 //Emacs Local Variables:
800 //Emacs mode:c++
801 //Emacs c-basic-offset:3
802 //Emacs tab-width:8
803 //Emacs indent-tabs-mode:nil
804 //Emacs End:
805 //-----------------------------------------------------------------------------