SoPlex Doxygen Documentation
spxchangebasis.cpp
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 //#define DEBUGGING 1
17 
18 #include <iostream>
19 
20 #include "spxdefines.h"
21 #include "spxbasis.h"
22 #include "spxsolver.h"
23 #include "spxout.h"
24 #include "exceptions.h"
25 
26 namespace soplex
27 {
28 
30 {
31  METHOD( "SPxBasis::reDim()" );
32 
33  assert(theLP != 0);
34 
35  MSG_DEBUG( spxout << "DCHBAS01 SPxBasis::reDim():"
36  << " matrixIsSetup=" << matrixIsSetup
37  << " fatorized=" << factorized
38  << std::endl; )
39 
41 
42  if (theLP->dim() != matrix.size())
43  {
44  MSG_INFO3( spxout << "ICHBAS02 basis redimensioning invalidates factorization"
45  << std::endl; )
46 
47  matrix.reSize (theLP->dim());
48  theBaseId.reSize(theLP->dim());
49  matrixIsSetup = false;
50  factorized = false;
51  }
52 
53  MSG_DEBUG( spxout << "DCHBAS03 SPxBasis::reDim(): -->"
54  << " matrixIsSetup=" << matrixIsSetup
55  << " fatorized=" << factorized
56  << std::endl; )
57 
58  assert( matrix.size() >= theLP->dim() );
59  assert( theBaseId.size() >= theLP->dim() );
60 }
61 
63 {
64  METHOD( "SPxBasis::addedRows()" );
65 
66  assert(theLP != 0);
67 
68  if( n > 0 )
69  {
70  reDim();
71 
72  if (theLP->rep() == SPxSolver::COLUMN)
73  {
74  /* I think, after adding rows in column representation,
75  reDim should set these bools to false. */
76  assert( !matrixIsSetup && !factorized );
77 
78  for (int i = theLP->nRows() - n; i < theLP->nRows(); ++i)
79  {
81  baseId(i) = theLP->SPxLP::rId(i);
82  }
83 
84  /* ??? I think, this cannot happen. */
85  /* if matrix was set up, load new basis vectors to the matrix */
86  if (status() > NO_PROBLEM && matrixIsSetup)
88  }
89  else
90  {
91  assert(theLP->rep() == SPxSolver::ROW);
92 
93  for (int i = theLP->nRows() - n; i < theLP->nRows(); ++i)
95 
96  /* In the row case, the basis is not effected by adding rows. However,
97  * since @c matrix stores references to the rows in the LP (SPxLP), a realloc
98  * in SPxLP (e.g. due to space requirements) might invalidate these references.
99  * We therefore have to "reload" the matrix if it is set up. Note that reDim()
100  * leaves @c matrixIsSetup untouched if only row have been added, since the basis
101  * matrix already has the correct size. */
102  if (status() > NO_PROBLEM && matrixIsSetup)
103  loadMatrixVecs();
104  }
105 
106  /* update basis status */
107  switch (status())
108  {
109  case PRIMAL:
110  case UNBOUNDED:
112  break;
113  case OPTIMAL:
114  case INFEASIBLE:
115  setStatus(DUAL);
116  break;
117  case NO_PROBLEM:
118  case SINGULAR:
119  case REGULAR:
120  case DUAL:
121  break;
122  default:
123  MSG_ERROR( spxout << "ECHBAS04 Unknown basis status!" << std::endl; )
124  throw SPxInternalCodeException("XCHBAS01 This should never happen.");
125  }
126  }
127 }
128 
130 {
131  METHOD( "SPxBasis::removedRow()" );
132 
133  assert(status() > NO_PROBLEM);
134  assert(theLP != 0);
135 
136  if (theLP->rep() == SPxSolver::ROW)
137  {
138  if (theLP->isBasic(thedesc.rowStatus(i)))
139  {
141  factorized = false;
142 
143  MSG_DEBUG( spxout << "DCHBAS05 Warning: deleting basic row!\n"; )
144  }
145  }
146  else
147  {
148  assert(theLP->rep() == SPxSolver::COLUMN);
149  factorized = false;
150  if (!theLP->isBasic(thedesc.rowStatus(i)))
151  {
153  MSG_DEBUG( spxout << "DCHBAS06 Warning: deleting nonbasic row!\n"; )
154  }
155  else if (status() > NO_PROBLEM && matrixIsSetup)
156  {
157  for (int j = theLP->dim(); j >= 0; --j)
158  {
159  SPxId id = baseId(j);
160 
161  if (id.isSPxRowId() && theLP->number(SPxRowId(id)) < 0)
162  {
163  baseId(j) = baseId(theLP->dim());
164 
165  if (j < theLP->dim())
166  matrix[j] = &theLP->vector(baseId(j));
167  break;
168  }
169  }
170  }
171  }
173  reDim();
174 }
175 
176 void SPxBasis::removedRows(const int perm[])
177 {
178  METHOD( "SPxBasis::removedRows()" );
179  assert(status() > NO_PROBLEM);
180  assert(theLP != 0);
181 
182  int i;
183  int n = thedesc.nRows();
184 
185  if (theLP->rep() == SPxSolver::ROW)
186  {
187  for (i = 0; i < n; ++i)
188  {
189  if (perm[i] != i)
190  {
191  if (perm[i] < 0) // row got removed
192  {
193  if (theLP->isBasic(thedesc.rowStatus(i)))
194  {
196  factorized = matrixIsSetup = false;
197  MSG_DEBUG( spxout << "DCHBAS07 Warning: deleting basic row!\n"; )
198  }
199  }
200  else // row was moved
201  thedesc.rowStatus(perm[i]) = thedesc.rowStatus(i);
202  }
203  }
204  }
205  else
206  {
207  assert(theLP->rep() == SPxSolver::COLUMN);
208 
209  factorized = false;
210  matrixIsSetup = false;
211 
212  for (i = 0; i < n; ++i)
213  {
214  if (perm[i] != i)
215  {
216  if (perm[i] < 0) // row got removed
217  {
218  if (!theLP->isBasic(thedesc.rowStatus(i)))
220  }
221  else // row was moved
222  thedesc.rowStatus(perm[i]) = thedesc.rowStatus(i);
223  }
224  }
225  }
226  reDim();
227 }
228 
229 
232 {
233  assert(theLP != 0);
234 
235  if (theLP->upper(i) < infinity)
236  {
237  if (theLP->lower(i) > -infinity)
238  {
239  if (theLP->lower(i) == theLP->SPxLP::upper(i))
241  /*
242  else
243  return (-theLP->lower(i) < theLP->upper(i))
244  ? SPxBasis::Desc::P_ON_LOWER
245  : SPxBasis::Desc::P_ON_UPPER;
246  */
247  else if (theLP->maxObj(i) == 0)
248  return (-theLP->lower(i) < theLP->upper(i))
251  else
252  return (theLP->maxObj(i) < 0)
255  }
256  else
258  }
259  else if (theLP->lower(i) > -infinity)
261  else
262  return SPxBasis::Desc::P_FREE;
263 }
264 
265 
267 {
268  METHOD( "SPxBasis::addedCols()" );
269  assert(theLP != 0);
270 
271  if( n > 0 )
272  {
273  reDim();
274 
275  if (theLP->rep() == SPxSolver::ROW)
276  {
277  /* I think, after adding columns in row representation,
278  reDim should set these bools to false. */
279  assert( !matrixIsSetup && !factorized );
280 
281  for (int i = theLP->nCols() - n; i < theLP->nCols(); ++i)
282  {
284  baseId(i) = theLP->SPxLP::cId(i);
285  }
286 
287  /* ??? I think, this cannot happen. */
288  /* if matrix was set up, load new basis vectors to the matrix */
289  if (status() > NO_PROBLEM && matrixIsSetup)
290  loadMatrixVecs();
291  }
292  else
293  {
294  assert(theLP->rep() == SPxSolver::COLUMN);
295 
296  for (int i = theLP->nCols() - n; i < theLP->nCols(); ++i)
298 
299  /* In the column case, the basis is not effected by adding columns. However,
300  * since @c matrix stores references to the columns in the LP (SPxLP), a realloc
301  * in SPxLP (e.g. due to space requirements) might invalidate these references.
302  * We therefore have to "reload" the matrix if it is set up. Note that reDim()
303  * leaves @c matrixIsSetup untouched if only columns have been added, since the
304  * basis matrix already has the correct size. */
305  if (status() > NO_PROBLEM && matrixIsSetup)
306  loadMatrixVecs();
307  }
308 
309  switch (status())
310  {
311  case DUAL:
312  case INFEASIBLE:
314  break;
315  case OPTIMAL:
316  case UNBOUNDED:
317  setStatus(PRIMAL);
318  break;
319  case NO_PROBLEM:
320  case SINGULAR:
321  case REGULAR:
322  case PRIMAL:
323  break;
324  default:
325  MSG_ERROR( spxout << "ECHBAS08 Unknown basis status!" << std::endl; )
326  throw SPxInternalCodeException("XCHBAS02 This should never happen.");
327  }
328  }
329 }
330 
332 {
333  METHOD( "SPxBasis::removedCol()" );
334  assert(status() > NO_PROBLEM);
335  assert(theLP != 0);
336 
337  if (theLP->rep() == SPxSolver::COLUMN)
338  {
339  if (theLP->isBasic(thedesc.colStatus(i)))
341  }
342  else
343  {
344  assert(theLP->rep() == SPxSolver::ROW);
345  factorized = false;
346  if (!theLP->isBasic(thedesc.colStatus(i)))
348  else if (status() > NO_PROBLEM)
349  {
350  for (int j = theLP->dim(); j >= 0; --j)
351  {
352  SPxId id = baseId(j);
353  if (id.isSPxColId()
354  && theLP->number(SPxColId(id)) < 0)
355  {
356  baseId(j) = baseId(theLP->dim());
357  if ( matrixIsSetup &&
358  j < theLP->dim() )
359  matrix[j] = &theLP->vector(baseId(j));
360  break;
361  }
362  }
363  }
364  }
365 
367  reDim();
368 }
369 
370 void SPxBasis::removedCols(const int perm[])
371 {
372  METHOD( "SPxBasis::removedCols()" );
373  assert(status() > NO_PROBLEM);
374  assert(theLP != 0);
375 
376  int i;
377  int n = thedesc.nCols();
378 
379  if (theLP->rep() == SPxSolver::COLUMN)
380  {
381  for (i = 0; i < n; ++i)
382  {
383  if (perm[i] < 0) // column got removed
384  {
385  if (theLP->isBasic(thedesc.colStatus(i)))
387  }
388  else // column was potentially moved
389  thedesc.colStatus(perm[i]) = thedesc.colStatus(i);
390  }
391  }
392  else
393  {
394  assert(theLP->rep() == SPxSolver::ROW);
395  factorized = matrixIsSetup = false;
396  for (i = 0; i < n; ++i)
397  {
398  if (perm[i] != i)
399  {
400  if (perm[i] < 0) // column got removed
401  {
402  if (!theLP->isBasic(thedesc.colStatus(i)))
404  }
405  else // column was moved
406  thedesc.colStatus(perm[i]) = thedesc.colStatus(i);
407  }
408  }
409  }
410 
411  reDim();
412 }
413 
414 
415 /**@todo is this correctly implemented?
416  */
418 {
419  METHOD( "SPxBasis::invalidate()" );
420 
421  MSG_INFO3( spxout << "ICHBAS09 explicit invalidation of factorization"
422  << std::endl; )
423 
424  factorized = false;
425  matrixIsSetup = false;
426 }
427 
428 
429 /**
430  This code has been adapted from SPxBasis::addedRows() and
431  SPxBasis::addedCols().
432 */
434 {
435  METHOD( "SPxBasis::restoreInitialBasis()" );
436 
437  assert( !matrixIsSetup && !factorized );
438 
439  if (theLP->rep() == SPxSolver::COLUMN)
440  {
441  for (int i = 0; i < theLP->nRows(); ++i)
442  {
444  baseId(i) = theLP->SPxLP::rId(i);
445  }
446 
447  for (int i = 0; i < theLP->nCols(); ++i)
449 
450  /* ??? I think, this cannot happen. */
451  /* if matrix was set up, load new basis vectors to the matrix */
452  if (status() > NO_PROBLEM && matrixIsSetup)
453  loadMatrixVecs();
454  }
455  else
456  {
457  assert(theLP->rep() == SPxSolver::ROW);
458 
459  for (int i = 0; i < theLP->nRows(); ++i)
461 
462  for (int i = 0; i < theLP->nCols(); ++i)
463  {
465  baseId(i) = theLP->SPxLP::cId(i);
466  }
467 
468  /* ??? I think, this cannot happen. */
469  /* if matrix was set up, load new basis vectors to the matrix */
470  if (status() > NO_PROBLEM && matrixIsSetup)
471  loadMatrixVecs();
472  }
473 
474  /* update basis status */
476 }
477 
478 /**
479  * @todo Implement changedRow(), changedCol(), changedElement() in a more clever
480  * way. For instance, the basis won't be singular (but maybe infeasible) if the
481  * change doesn't affect the basis rows/columns.
482  *
483  * The following methods (changedRow(), changedCol(), changedElement()) radically
484  * change the current basis to the original (slack) basis also present after
485  * loading the LP. The reason is that through the changes, the current basis may
486  * become singular. Going back to the initial basis is quite inefficient, but
487  * correct.
488  */
489 
490 /**@todo is this correctly implemented?
491  */
492 void SPxBasis::changedRow(int /*row*/)
493 {
494  METHOD( "SPxBasis::changedRow()" );
495  invalidate();
497 }
498 
499 /**@todo is this correctly implemented?
500  */
501 void SPxBasis::changedCol(int /*col*/)
502 {
503  METHOD( "SPxBasis::changedCol()" );
504  invalidate();
506 }
507 
508 /**@todo is this correctly implemented?
509  */
510 void SPxBasis::changedElement(int /*row*/, int /*col*/)
511 {
512  METHOD( "SPxBasis::changedElement()" );
513  invalidate();
515 }
516 } // namespace soplex
517 
518 //-----------------------------------------------------------------------------
519 //Emacs Local Variables:
520 //Emacs mode:c++
521 //Emacs c-basic-offset:3
522 //Emacs tab-width:8
523 //Emacs indent-tabs-mode:nil
524 //Emacs End:
525 //-----------------------------------------------------------------------------