SoPlex Doxygen Documentation
changesoplex.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 <assert.h>
19 #include <iostream>
20 
21 #include "spxdefines.h"
22 #include "spxsolver.h"
23 #include "spxpricer.h"
24 #include "spxratiotester.h"
25 #include "exceptions.h"
26 
27 namespace soplex
28 {
29 
30 #if 0
31 void SPxSolver::localAddRows(int start)
32 {
33  METHOD( "SPxSolver::localAddRows()" );
34  assert( start <= SPxLP::nRows() );
35 
36  /**@todo This method seems to be called, to update
37  * theFvec, theFrhs, ..., but a resolve after
38  * adding a row results in a failure.
39  * To fix this, we call unInit() so that init() is called before solving
40  * in spxsolve.cpp:solve(). In init(), the
41  * vectors are set up, so there is no need
42  * to update them here.
43  */
44  if( start == SPxLP::nRows() )
45  return;
46 
47  const SPxBasis::Desc& ds = desc();
48 
49  if (type() == ENTER)
50  {
51  if (rep() == COLUMN)
52  {
53  int i;
54  for (i = start; i < SPxLP::nRows(); ++i)
55  {
56  theURbound[i] = -lhs(i);
57  theLRbound[i] = -rhs(i);
58  setEnterBound4Row(i, i);
60  // init #theFrhs[i]#:
61  Real& v_rhs = (*theFrhs)[i];
62  const SVector& row = rowVector(i);
63  v_rhs = 0;
64  for (int j = row.size() - 1; j >= 0; --j)
65  {
66  int idx = row.index(j);
67  switch (ds.colStatus(idx))
68  {
69  case Desc::P_ON_UPPER:
70  v_rhs += row.value(j) * theUCbound[idx];
71  break;
72  case Desc::P_ON_LOWER:
73  case Desc::P_FIXED:
74  v_rhs += row.value(j) * theLCbound[idx];
75  break;
76  default:
77  break;
78  }
79  }
80  }
83  for (i = start; i < SPxLP::nRows(); ++i)
84  {
85  /* we need to compare with tolerance (rep() == COLUMN) ? feastol() : opttol() because theFvec is the primal
86  * vector in COLUMN and the dual vector in ROW representation; this is equivalent to entertol(); this also
87  * fits because we are within the "type() == ENTER" case
88  */
89  if (theUBbound[i] + entertol() < (*theFvec)[i])
90  shiftUBbound(i, (*theFvec)[i]);
91  else if ((*theFvec)[i] < theLBbound[i] - entertol())
92  shiftLBbound(i, (*theFvec)[i]);
93  }
94  computePvec();
95  computeCoTest();
96  computeTest();
97  }
98  else
99  {
100  assert(rep() == ROW);
101  for (int i = start; i < SPxLP::nRows(); ++i)
102  {
103  theURbound[i] = theLRbound[i] = 0;
105  theURbound[i], theLRbound[i]);
106  (*thePvec)[i] = vector(i) * (*theCoPvec);
107  theTest[i] = test(i, ds.status(i));
108  }
109  }
110  }
111  else
112  {
113  assert(type() == LEAVE);
114  if (rep() == ROW)
115  {
116  for (int i = start; i < SPxLP::nRows(); ++i)
117  {
118  theURbound[i] = rhs(i);
119  theLRbound[i] = lhs(i);
120  (*thePvec)[i] = vector(i) * (*theCoPvec);
121 
122  /* we need to compare with tolerance (rep() == ROW) ? feastol() : opttol() because thePvec is the primal
123  * vector in ROW and the dual vector in COLUMN representation; this is equivalent to leavetol(); this also
124  * fits because we are within the "type() == LEAVE" case
125  */
126  if (theURbound[i] + leavetol() < (*thePvec)[i])
127  shiftUPbound(i, (*thePvec)[i]);
128  else if ((*thePvec)[i] < theLRbound[i] - leavetol())
129  shiftLPbound(i, (*thePvec)[i]);
130  }
131  }
132  else
133  {
134  assert(rep() == COLUMN);
135  int i;
136  for (i = start; i < SPxLP::nRows(); ++i)
137  {
138  theURbound[i] = theLRbound[i] = 0;
139  clearDualBounds(ds.rowStatus(i),
140  theURbound[i], theLRbound[i]);
141  setLeaveBound4Row(i, i);
143  // init #theFrhs[i]#
144  Real& v_rhs = (*theFrhs)[i];
145  const SVector& row = rowVector(i);
146  v_rhs = 0;
147  for (int j = row.size() - 1; j >= 0; --j)
148  {
149  int idx = row.index(j);
150  switch (ds.colStatus(idx))
151  {
152  case Desc::P_ON_UPPER:
153  v_rhs += row.value(j) * SPxLP::upper(idx);
154  break;
155  case Desc::P_ON_LOWER:
156  case Desc::P_FIXED:
157  v_rhs += row.value(j) * SPxLP::lower(idx);
158  break;
159  default:
160  break;
161  }
162  }
163  }
166  for (i = start; i < SPxLP::nRows(); ++i)
167  {
168  if ((*theFvec)[i] > theUBbound[i])
169  theCoTest[i] = theUBbound[i] - (*theFvec)[i];
170  else
171  theCoTest[i] = (*theFvec)[i] - theLBbound[i];
172  }
173  }
174  }
175 }
176 
177 void SPxSolver::addedRows(int n)
178 {
179  METHOD( "SPxSolver::addedRows()" );
180 
181  SPxLP::addedRows(n);
182 
183  if( n > 0 )
184  {
185  reDim();
186 
188  {
190 
191  if (isInitialized())
192  {
193  localAddRows(nRows() - n);
194 
195  assert(thepricer != 0);
196 
197  if (rep() == ROW)
198  thepricer->addedVecs(n);
199  else
200  thepricer->addedCoVecs(n);
201  }
202  }
203  }
204 
205  /* we must not assert consistency here, since addedCols() might be still necessary to obtain a consistent basis */
206 }
207 #endif //0
208 
210 {
211  METHOD( "SPxSolver::addedRows()" );
212 
213  if( n > 0 )
214  {
215  SPxLP::addedRows(n);
216 
217  unInit();
218  reDim();
219 
222  }
223 
224  /* we must not assert consistency here, since addedCols() might be still necessary to obtain a consistent basis */
225 }
226 
227 #if 0
228 void SPxSolver::localAddCols(int start)
229 {
230  METHOD( "SPxSolver::localAddCols()" );
231  assert( start <= SPxLP::nCols() );
232 
233  /**@todo This method seems to be called, to update
234  * theFvec, theFrhs, ..., but a resolve after
235  * adding a row results in a failure.
236  * To fix this, we call unIinit() so that init() is called before solving
237  * in spxsolve.cpp:solve(). In init(), the
238  * vectors are set up, so there is no need
239  * to update them here.
240  */
241  if( start == SPxLP::nCols() )
242  return;
243 
244  const SPxBasis::Desc& ds = desc();
245 
246  if (type() == ENTER)
247  {
248  if (rep() == COLUMN)
249  {
250  int reSolve = 0;
251  int i;
252  Real x;
253  for (i = start; i < SPxLP::nCols(); ++i)
254  {
255  (*thePvec)[i] = vector(i) * (*theCoPvec);
256  theTest[i] = test(i, ds.colStatus(i));
257  theUCbound[i] = SPxLP::upper(i);
258  theLCbound[i] = SPxLP::lower(i);
259  switch (ds.colStatus(i))
260  {
262  assert(SPxLP::lower(i) == SPxLP::upper(i));
263  /*FALLTHROUGH*/
265  x = SPxLP::upper(i);
266  break;
268  x = SPxLP::lower(i);
269  break;
270  default:
271  x = 0;
272  break;
273  }
274  if (x)
275  {
276  theFrhs->multAdd(-x, vector(i));
277  reSolve = 1;
278  }
279  }
280  if (reSolve)
281  {
283  shiftFvec();
284  }
285  }
286  else
287  {
288  int i;
289  for (i = start; i < SPxLP::nCols(); ++i)
290  {
291  theUCbound[i] = theLCbound[i] = 0;
292  (*theFrhs)[i] = SPxLP::spxSense() * SPxLP::obj(i);
294  theUCbound[i], theLCbound[i]);
295  setEnterBound4Col(i, i);
297  }
299  computePvec();
300  computeCoTest();
301  computeTest();
303  for (i = start; i < SPxLP::nCols(); ++i)
304  {
305  /* we need to compare with tolerance (rep() == COLUMN) ? feastol() : opttol() because theFvec is the primal
306  * vector in COLUMN and the dual vector in ROW representation; this is equivalent to entertol(); this also
307  * fits because we are within the "type() == ENTER" case
308  */
309  if (theUBbound[i] + entertol() < (*theFvec)[i])
310  shiftUBbound(i, (*theFvec)[i]);
311  if ((*theFvec)[i] < theLBbound[i] - entertol())
312  shiftLBbound(i, (*theFvec)[i]);
313  }
314  }
315  }
316  else
317  {
318  if (rep() == ROW)
319  {
320  int i;
321  for (i = start; i < SPxLP::nCols(); ++i)
322  {
323  theUCbound[i] = SPxLP::upper(i);
324  theLCbound[i] = SPxLP::lower(i);
325  (*theFrhs)[i] = SPxLP::spxSense() * SPxLP::obj(i);
326  setLeaveBound4Col(i, i);
328  }
330  computePvec();
331  // shiftPvec();
333  for (i = start; i < SPxLP::nCols(); ++i)
334  {
335  if ((*theFvec)[i] > theUBbound[i])
336  theCoTest[i] = theUBbound[i] - (*theFvec)[i];
337  else
338  theCoTest[i] = (*theFvec)[i] - theLBbound[i];
339  }
340  }
341  else
342  {
343  Real x;
344  int i;
345  int reSolve = 0;
346  for (i = start; i < SPxLP::nCols(); ++i)
347  {
348  theUCbound[i] = theLCbound[i] = -maxObj(i);
350  theLCbound[i], theUCbound[i]);
351  theUCbound[i] *= -1;
352  theLCbound[i] *= -1;
353 
354  (*thePvec)[i] = vector(i) * (*theCoPvec);
355 
356  /* we need to compare with tolerance (rep() == ROW) ? feastol() : opttol() because thePvec is the primal
357  * vector in ROW and the dual vector in COLUMN representation; this is equivalent to leavetol(); this also
358  * fits because we are within the "type() == LEAVE" case
359  */
360  if (theUCbound[i] + leavetol() < (*thePvec)[i])
361  shiftUPbound(i, (*thePvec)[i]);
362  if (theLCbound[i] - leavetol() > (*thePvec)[i])
363  shiftLPbound(i, (*thePvec)[i]);
364 
365  switch (ds.colStatus(i))
366  {
368  assert(SPxLP::lower(i) == SPxLP::upper(i));
369  /*FALLTHROUGH*/
371  x = SPxLP::upper(i);
372  break;
374  x = SPxLP::lower(i);
375  break;
376  default:
377  x = 0;
378  break;
379  }
380  if (x)
381  {
382  theFrhs->multAdd(-x, vector(i));
383  reSolve = 1;
384  }
385  }
386  if (reSolve)
387  {
389  computeFtest();
390  }
391  }
392  }
393 }
394 
395 void SPxSolver::addedCols(int n)
396 {
397  METHOD( "SPxSolver::addedCols()" );
398  SPxLP::addedCols(n);
399 
400  if( n > 0 )
401  {
402  reDim();
403 
405  {
407  if (isInitialized())
408  {
409  localAddCols(nCols() - n);
410  assert(thepricer != 0);
411  if (rep() == COLUMN)
412  thepricer->addedVecs(n);
413  else
415  }
416  }
417  }
418 
419  /* we must not assert consistency here, since addedRows() might be still necessary to obtain a consistent basis */
420 }
421 #endif //0
422 
424 {
425  METHOD( "SPxSolver::addedCols()" );
426 
427  if( n > 0 )
428  {
429  SPxLP::addedCols(n);
430 
431  unInit();
432  reDim();
433 
436  }
437 
438  /* we must not assert consistency here, since addedRows() might be still necessary to obtain a consistent basis */
439 }
440 
442 {
443  METHOD( "SPxSolver::doRemoveRow()" );
444 
446 
447  unInit();
448 
450  {
451  removedRow(i);
452 
453 #if 0
454  if (isInitialized())
455  {
456  int n = SPxLP::nRows();
457 
458  theURbound[i] = theURbound[n];
459  theLRbound[i] = theLRbound[n];
460 
461  if (rep() == ROW)
462  {
463  (*thePvec)[i] = (*thePvec)[n];
464  if (type() == ENTER)
465  theTest[i] = theTest[n];
466  reDim();
467  assert(thepricer != 0);
468  thepricer->removedVec(i);
469  }
470  else
471  {
472  unInit();
473  }
474  }
475 #endif // 0
476 
477  switch (SPxBasis::status())
478  {
479  case SPxBasis::DUAL:
482  break;
483  case SPxBasis::OPTIMAL:
485  break;
486  default:
487  break;
488  }
489  }
490 }
491 
492 void SPxSolver::doRemoveRows(int perm[])
493 {
494  METHOD( "SPxSolver::doRemoveRows()" );
495 
496  SPxLP::doRemoveRows(perm);
497 
498  unInit();
499 
501  {
502  removedRows(perm);
503 #if 0
504  if (isInitialized())
505  {
506  int n = SPxLP::nRows();
507 
508  if (rep() == ROW)
509  {
510  if (type() == ENTER)
511  {
512  for (int i = 0; i < n; ++i)
513  if (perm[i] >= 0)
514  {
515  theURbound[perm[i]] = theURbound[i];
516  theLRbound[perm[i]] = theLRbound[i];
517  (*thePvec)[perm[i]] = (*thePvec)[i];
518  theTest[perm[i]] = theTest[i];
519  }
520  }
521  else
522  {
523  for (int i = 0; i < n; ++i)
524  if (perm[i] >= 0)
525  {
526  theURbound[perm[i]] = theURbound[i];
527  theLRbound[perm[i]] = theLRbound[i];
528  (*thePvec)[perm[i]] = (*thePvec)[i];
529  }
530  }
531  assert(thepricer != 0);
532  thepricer->removedVecs(perm);
533  reDim();
534  }
535  else
536  {
537  unInit();
538  }
539  }
540 #endif
541  switch (SPxBasis::status())
542  {
543  case SPxBasis::DUAL:
546  break;
547  case SPxBasis::OPTIMAL:
549  break;
550  default:
551  break;
552  }
553  }
554 }
555 
557 {
558  METHOD( "SPxSolver::doRemoveCol()" );
559 
561 
562  unInit();
563 
565  {
566  removedCol(i);
567 
568 #if 0
569  if (isInitialized())
570  {
571  int n = SPxLP::nCols();
572 
573  theUCbound[i] = theUCbound[n];
574  theLCbound[i] = theLCbound[n];
575  if (rep() == COLUMN)
576  {
577  (*thePvec)[i] = (*thePvec)[n];
578  if (type() == ENTER)
579  theTest[i] = theTest[n];
580  assert(thepricer != 0);
581  thepricer->removedVec(i);
582  reDim();
583  }
584  else
585  {
586  unInit();
587  }
588  }
589 #endif //0
590  switch (SPxBasis::status())
591  {
592  case SPxBasis::PRIMAL:
593  case SPxBasis::UNBOUNDED:
595  break;
596  case SPxBasis::OPTIMAL:
598  break;
599  default:
600  break;
601  }
602  }
603 }
604 
605 void SPxSolver::doRemoveCols(int perm[])
606 {
607  METHOD( "SPxSolver::doRemoveCols()" );
608 
609  SPxLP::doRemoveCols(perm);
610 
611  unInit();
612 
614  {
615  removedCols(perm);
616 
617 #if 0
618  if (isInitialized())
619  {
620  int n = SPxLP::nCols();
621 
622  if (rep() == COLUMN)
623  {
624  if (type() == ENTER)
625  {
626  for (int i = 0; i < n; ++i)
627  if (perm[i] >= 0)
628  {
629  theUCbound[perm[i]] = theUCbound[i];
630  theLCbound[perm[i]] = theLCbound[i];
631  (*thePvec)[perm[i]] = (*thePvec)[i];
632  theTest[perm[i]] = theTest[i];
633  }
634  }
635  else
636  {
637  for (int i = 0; i < n; ++i)
638  if (perm[i] >= 0)
639  {
640  theUCbound[perm[i]] = theUCbound[i];
641  theLCbound[perm[i]] = theLCbound[i];
642  (*thePvec)[perm[i]] = (*thePvec)[i];
643  }
644  }
645  assert(thepricer != 0);
646  thepricer->removedVecs(perm);
647  reDim();
648  }
649  else
650  {
651  unInit();
652  }
653  }
654 #endif //0
655  switch (SPxBasis::status())
656  {
657  case SPxBasis::PRIMAL:
658  case SPxBasis::UNBOUNDED:
660  break;
661  case SPxBasis::OPTIMAL:
663  break;
664  default:
665  break;
666  }
667  }
668 }
669 
670 void SPxSolver::changeObj(const Vector& newObj)
671 {
672  METHOD( "SPxSolver::changeObj()" );
673 
674  SPxLP::changeObj(newObj);
675 
676  /**@todo Factorization remains valid, we do not need a reDim()
677  * pricing vectors should be recomputed.
678  */
679  unInit();
680 }
681 
682 void SPxSolver::changeObj(int i, Real newVal)
683 {
684  METHOD( "SPxSolver::changeObj()" );
685 
686  SPxLP::changeObj(i, newVal);
687 
688  /**@todo Factorization remains valid, we do not need a reDim()
689  * pricing vectors should be recomputed.
690  */
691  unInit();
692 }
693 
694 static void changeLowerStatus(
696  Real newLower,
697  Real upper,
698  const SPxBasis& basis,
699  int i)
700 {
701  MSG_DEBUG( spxout << "DCHANG01 changeLowerStatus(): col " << i
702  << "[" << newLower << ":" << upper << "] " << stat; )
703 
704  switch (stat)
705  {
707  if (newLower <= -infinity)
709  else if (newLower == upper)
711  break;
713  if (newLower == upper)
715  break;
717  if (newLower > -infinity)
719  break;
721  if (newLower != upper)
723  break;
729  stat = basis.dualColStatus(i);
730  break;
731  default:
732  throw SPxInternalCodeException("XCHANG01 This should never happen.");
733  }
734  MSG_DEBUG( spxout << " -> " << stat << std::endl; )
735 }
736 
737 void SPxSolver::changeLower(const Vector& newLower)
738 {
739  METHOD( "SPxSolver::changeLower()" );
740 
741  SPxLP::changeLower(newLower);
742 
744  {
745  for (int i = 0; i < newLower.dim(); ++i)
746  changeLowerStatus(desc().colStatus(i), newLower[i], upper(i), *this, i);
747 
748  unInit();
749  }
750 }
751 
752 void SPxSolver::changeLower(int i, Real newLower)
753 {
754  METHOD( "SPxSolver::changeLower()" );
755 
756  if (newLower != lower(i))
757  {
758  // This has to be done before calling changeLowerStatus() because that is calling
759  // basis.dualColStatus() which calls lower() and needs the changed value.
760  SPxLP::changeLower(i, newLower);
761 
763  {
764  changeLowerStatus(desc().colStatus(i), newLower, upper(i), *this, i);
765  unInit();
766  }
767  }
768 }
769 
770 static void changeUpperStatus(
772  Real newUpper,
773  Real lower,
774  const SPxBasis& basis,
775  int i)
776 {
777  MSG_DEBUG( spxout << "DCHANG02 changeUpperStatus(): col " << i
778  << "[" << lower << ":" << newUpper << "] " << stat; )
779 
780  switch (stat)
781  {
783  if (newUpper == lower)
785  break;
787  if (newUpper >= infinity)
789  else if (newUpper == lower)
791  break;
793  if (newUpper < infinity)
795  break;
797  if (newUpper != lower)
799  break;
805  stat = basis.dualColStatus(i);
806  break;
807  default:
808  throw SPxInternalCodeException("XCHANG02 This should never happen.");
809  }
810  MSG_DEBUG( spxout << " -> " << stat << std::endl; );
811 }
812 
813 void SPxSolver::changeUpper(const Vector& newUpper)
814 {
815  METHOD( "SPxSolver::changeUpper()" );
816 
817  SPxLP::changeUpper(newUpper);
818 
820  {
821  for (int i = 0; i < newUpper.dim(); ++i)
822  changeUpperStatus(desc().colStatus(i), newUpper[i], lower(i), *this, i);
823 
824  unInit();
825  }
826 }
827 
828 void SPxSolver::changeUpper(int i, Real newUpper)
829 {
830  METHOD( "SPxSolver::changeUpper()" );
831 
832  if (newUpper != upper(i))
833  {
834  SPxLP::changeUpper(i, newUpper);
835 
837  {
838  changeUpperStatus(desc().colStatus(i), newUpper, lower(i), *this, i);
839  unInit();
840  }
841  }
842 }
843 
844 void SPxSolver::changeBounds(const Vector& newLower, const Vector& newUpper)
845 {
846  METHOD( "SPxSolver::changeBounds()" );
847 
848  changeLower(newLower);
849  changeUpper(newUpper);
850 }
851 
852 void SPxSolver::changeBounds(int i, Real newLower, Real newUpper)
853 {
854  METHOD( "SPxSolver::changeBounds()" );
855 
856  changeLower(i, newLower);
857  changeUpper(i, newUpper);
858 }
859 
860 /**@todo Change Lhs/Rhs Status the same way as changeBounds
861  */
862 static void changeLhsStatus(
864  Real newLhs,
865  Real rhs,
866  const SPxBasis& basis,
867  int i)
868 {
869  MSG_DEBUG( spxout << "DCHANG03 changeLhsStatus() : row " << i
870  << ": " << stat; )
871  switch (stat)
872  {
874  if (newLhs <= -infinity)
876  else if (newLhs == rhs)
878  break;
880  if (newLhs == rhs)
882  break;
884  if (newLhs > -infinity)
886  break;
888  if (newLhs != rhs)
890  break;
896  stat = basis.dualRowStatus(i);
897  break;
898  default:
899  throw SPxInternalCodeException("XCHANG03 This should never happen.");
900  }
901  MSG_DEBUG( spxout << " -> " << stat << std::endl; )
902 }
903 
904 void SPxSolver::changeLhs(const Vector& newLhs)
905 {
906  METHOD( "SPxSolver::changeLhs()" );
907 
908  SPxLP::changeLhs(newLhs);
909 
911  {
912  for (int i = 0; i < nRows(); ++i)
913  changeLhsStatus(desc().rowStatus(i), newLhs[i], rhs(i), *this, i);
914 
915  unInit();
916  }
917 }
918 
919 void SPxSolver::changeLhs(int i, Real newLhs)
920 {
921  METHOD( "SPxSolver::changeLhs()" );
922 
923  SPxLP::changeLhs(i, newLhs);
924 
926  {
927  changeLhsStatus(desc().rowStatus(i), newLhs, rhs(i), *this, i);
928  unInit();
929  }
930 }
931 
932 static void changeRhsStatus(
934  Real newRhs,
935  Real lhs,
936  const SPxBasis& basis,
937  int i)
938 {
939  MSG_DEBUG( spxout << "DCHANG04 changeRhsStatus() : row " << i
940  << ": " << stat; )
941  switch (stat)
942  {
944  if (newRhs >= infinity)
946  else if (newRhs == lhs)
948  break;
950  if (newRhs == lhs)
952  break;
954  if (newRhs < infinity)
956  break;
958  if (newRhs != lhs)
960  break;
966  stat = basis.dualRowStatus(i);
967  break;
968  default:
969  throw SPxInternalCodeException("XCHANG04 This should never happen.");
970  }
971  MSG_DEBUG( spxout << " -> " << stat << std::endl; )
972 }
973 
974 
975 void SPxSolver::changeRhs(const Vector& newRhs)
976 {
977  METHOD( "SPxSolver::changeRhs()" );
978 
979  SPxLP::changeRhs(newRhs);
980 
982  {
983  for (int i = 0; i < nRows(); ++i)
984  changeRhsStatus(desc().rowStatus(i), newRhs[i], lhs(i), *this, i);
985  unInit();
986  }
987 }
988 
989 void SPxSolver::changeRhs(int i, Real newRhs)
990 {
991  METHOD( "SPxSolver::changeRhs()" );
992 
993  SPxLP::changeRhs(i, newRhs);
994 
996  {
997  changeRhsStatus(desc().rowStatus(i), newRhs, lhs(i), *this, i);
998  unInit();
999  }
1000 }
1001 
1002 void SPxSolver::changeRange(const Vector& newLhs, const Vector& newRhs)
1003 {
1004  METHOD( "SPxSolver::changeRange()" );
1005  SPxLP::changeLhs(newLhs);
1006  SPxLP::changeRhs(newRhs);
1008  {
1009  for (int i = nRows() - 1; i >= 0; --i)
1010  {
1011  changeLhsStatus(desc().rowStatus(i), newLhs[i], rhs(i), *this, i);
1012  changeRhsStatus(desc().rowStatus(i), newRhs[i], lhs(i), *this, i);
1013  }
1014  unInit();
1015  }
1016 }
1017 
1018 void SPxSolver::changeRange(int i, Real newLhs, Real newRhs)
1019 {
1020  METHOD( "SPxSolver::changeRange()" );
1021 
1022  SPxLP::changeLhs(i, newLhs);
1023  SPxLP::changeRhs(i, newRhs);
1024 
1026  {
1027  changeLhsStatus(desc().rowStatus(i), newLhs, rhs(i), *this, i);
1028  changeRhsStatus(desc().rowStatus(i), newRhs, lhs(i), *this, i);
1029  unInit();
1030  }
1031 }
1032 
1033 void SPxSolver::changeRow(int i, const LPRow& newRow)
1034 {
1035  METHOD( "SPxSolver::changeRow()" );
1036 
1037  SPxLP::changeRow(i, newRow);
1039  SPxBasis::changedRow( i );
1040  unInit();
1041 }
1042 
1043 void SPxSolver::changeCol(int i, const LPCol& newCol)
1044 {
1045  METHOD( "SPxSolver::changeCol()" );
1046 
1047  SPxLP::changeCol(i, newCol);
1049  SPxBasis::changedCol( i );
1050  unInit();
1051 }
1052 
1053 void SPxSolver::changeElement(int i, int j, Real val)
1054 {
1055  METHOD( "SPxSolver::changeElement()" );
1056 
1057  SPxLP::changeElement(i, j, val);
1059  SPxBasis::changedElement( i, j );
1060  unInit();
1061 }
1062 
1064 {
1065  METHOD( "SPxSolver::changeSense()" );
1066 
1067  SPxLP::changeSense(sns);
1068  unInit();
1069 }
1070 } // namespace soplex
1071 
1072 //-----------------------------------------------------------------------------
1073 //Emacs Local Variables:
1074 //Emacs mode:c++
1075 //Emacs c-basic-offset:3
1076 //Emacs tab-width:8
1077 //Emacs indent-tabs-mode:nil
1078 //Emacs End:
1079 //-----------------------------------------------------------------------------