SoPlex Doxygen Documentation
spxbounds.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 
24 namespace soplex
25 {
26 /** Setting up the feasiblity bound for normal primal variables is
27  straightforward. However, slack variables need some more details
28  on how we treate them. This is slightly different from usual
29  textbook versions. Let \f$l_i \le A_i^T x \le u_i\f$. This can be
30  transformed to \f$A_i^Tx + s_i = 0\f$, with \f$-u_i \le s_i \le
31  -l_i\f$. Hence, with this definition of slack variables \f$s_i\f$, we
32  can directly use vectors \f$l\f$ and \f$u\f$ as feasibility bounds.
33  */
35 {
36  METHOD( "SPxSolver::setPrimalBounds()" );
37 
40 
41  if (rep() == ROW)
42  {
43  theURbound = rhs();
44  theLRbound = lhs();
45  }
46  else
47  {
48  theURbound = lhs();
49  theLRbound = rhs();
50  theURbound *= -1.0;
51  theLRbound *= -1.0;
52  }
53 }
54 
55 /** Seting up the basis for dual simplex requires to install upper and lower
56  feasibility bounds for dual variables (|Lbound| and |Ubound|). Here is a
57  list of how these must be set for inequalities of type \f$l \le a^Tx \le u\f$:
58 
59  \f[
60  \begin{tabular}{cccc}
61  $l$ & $u$ & |Lbound| & |Ubound| \\
62  \hline
63  $-\infty=l$ & $u=\infty$ & 0 & 0 \\
64  $-\infty<l$ & $u=\infty$ & 0 & $\infty$ \\
65  $-\infty=l$ & $u<\infty$ & $-\infty$ & 0 \\
66  \multicolumn{2}{c}{
67  $-\infty<l \ne u<\infty$} & 0 & 0 \\
68  \multicolumn{2}{c}{
69  $-\infty<l = u<\infty$} & $-\infty$ & $\infty$ \\
70  \end{tabular}
71  \f]
72  The case \f$l = -\infty\f$, \f$u = \infty\f$ occurs for unbounded primal variables.
73  Such must be treated differently from the general case.
74 
75  Given possible upper and lower bounds to a dual variable with |Status stat|,
76  this function clears the bounds according to |stat| by setting them to
77  \f$\infty\f$ or \f$-\infty\f$, respectively.
78  */
81  Real& upp,
82  Real& lw) const
83 {
84  METHOD( "SPxSolver::clearDualBounds()" );
85 
86  switch (stat)
87  {
90  upp = infinity;
91  lw = -infinity;
92  break;
95  upp = infinity;
96  break;
99  lw = -infinity;
100  break;
101 
102  default:
103  break;
104  }
105 }
106 
108 {
109  METHOD( "SPxSolver::setDualColBounds()" );
110 
111  assert(rep() == COLUMN);
112 
113  const SPxBasis::Desc& ds = desc();
114 
115  int i;
116 
117  for(i = 0; i < nRows(); ++i)
118  {
119  theURbound[i] = 0.0;
120  theLRbound[i] = 0.0;
121 
123  }
124 
125  for(i = 0; i < nCols(); ++i)
126  {
127  theUCbound[i] = -maxObj(i);
128  theLCbound[i] = -maxObj(i);
129 
130  // exchanged ... due to definition of slacks!
132 
133  theUCbound[i] *= -1.0;
134  theLCbound[i] *= -1.0;
135  }
136 }
137 
139 {
140  METHOD( "SPxSolver::setDualRowBounds()" );
141 
142  assert(rep() == ROW);
143 
144  int i;
145 
146  for(i = 0; i < nRows(); ++i)
147  {
148  theURbound[i] = 0.0;
149  theLRbound[i] = 0.0;
150 
152  }
153 
154  for(i = 0; i < nCols(); ++i)
155  {
156  theUCbound[i] = 0.0;
157  theLCbound[i] = 0.0;
158 
160  }
161 }
162 
163 
164 /** This sets up the bounds for basic variables for entering simplex algorithms.
165  It requires, that all upper lower feasibility bounds have allready been
166  setup. Method |setEnterBound4Row(i, n)| does so for the |i|-th basis
167  variable being row index |n|. Equivalently, method
168  |setEnterBound4Col(i, n)| does so for the |i|-th basis variable being
169  column index |n|.
170  */
172 {
173  METHOD( "SPxSolver::setEnterBound4Row()" );
174  assert(baseId(i).isSPxRowId());
175  assert(number(SPxRowId(baseId(i))) == n);
176  switch (desc().rowStatus(n))
177  {
179  theLBbound[i] = -infinity;
180  theUBbound[i] = theURbound[n];
181  break;
183  theLBbound[i] = theLRbound[n];
184  theUBbound[i] = infinity;
185  break;
186 
187  default:
188  theUBbound[i] = theURbound[n];
189  theLBbound[i] = theLRbound[n];
190  break;
191  }
192 }
193 
195 {
196  METHOD( "SPxSolver::setEnterBound4Col()" );
197  assert(baseId(i).isSPxColId());
198  assert(number(SPxColId(baseId(i))) == n);
199  switch (desc().colStatus(n))
200  {
202  theLBbound[i] = -infinity;
203  theUBbound[i] = theUCbound[n];
204  break;
206  theLBbound[i] = theLCbound[n];
207  theUBbound[i] = infinity;
208  break;
209 
210  default:
211  theUBbound[i] = theUCbound[n];
212  theLBbound[i] = theLCbound[n];
213  break;
214  }
215 }
216 
218 {
219  METHOD( "SPxSolver::setEnterBounds()" );
220 
221  for (int i = 0; i < dim(); ++i)
222  {
223  SPxId base_id = baseId(i);
224 
225  if (base_id.isSPxRowId())
226  setEnterBound4Row(i, number(SPxRowId(base_id)));
227  else
228  setEnterBound4Col(i, number(SPxColId(base_id)));
229  }
230 }
231 
232 
233 /** This sets up the bounds for basic variables for leaving simplex algorithms.
234  While method |setLeaveBound4Row(i,n)| does so for the |i|-th basic variable
235  being the |n|-th row, |setLeaveBound4Col(i,n)| does so for the |i|-th basic
236  variable being the |n|-th column.
237  */
239 {
240  METHOD( "SPxSolver::setLeaveBound4Row()" );
241  assert(baseId(i).isSPxRowId());
242  assert(number(SPxRowId(baseId(i))) == n);
243  switch (desc().rowStatus(n))
244  {
246  theLBbound[i] = -infinity;
247  theUBbound[i] = 0.0;
248  break;
250  theLBbound[i] = 0.0;
251  theUBbound[i] = infinity;
252  break;
254  theLBbound[i] = -infinity;
255  theUBbound[i] = infinity;
256  break;
258  theLBbound[i] = 0.0;
259  theUBbound[i] = 0.0;
260  break;
261 
262  default:
263  assert(rep() == COLUMN);
264  theLBbound[i] = -rhs(n); // slacks !
265  theUBbound[i] = -lhs(n); // slacks !
266  break;
267  }
268 }
269 
271 {
272  METHOD( "SPxSolver::setLeaveBound4Col()" );
273 
274  assert(baseId(i).isSPxColId());
275  assert(number(SPxColId(baseId(i))) == n);
276 
277  switch (desc().colStatus(n))
278  {
280  theLBbound[i] = -infinity;
281  theUBbound[i] = 0;
282  break;
284  theLBbound[i] = 0;
285  theUBbound[i] = infinity;
286  break;
288  theLBbound[i] = -infinity;
289  theUBbound[i] = infinity;
290  break;
292  theLBbound[i] = theUBbound[i] = 0;
293  break;
294 
295  default:
296  theUBbound[i] = SPxLP::upper(n);
297  theLBbound[i] = SPxLP::lower(n);
298  break;
299  }
300 }
301 
303 {
304  METHOD( "SPxSolver::setLeaveBounds()" );
305 
306  for (int i = 0; i < dim(); ++i)
307  {
308  SPxId base_id = baseId(i);
309 
310  if (base_id.isSPxRowId())
311  setLeaveBound4Row(i, number(SPxRowId(base_id)));
312  else
313  setLeaveBound4Col(i, number(SPxColId(base_id)));
314  }
315 }
316 
318 {
319  METHOD( "SPxSolver::testBounds()" );
320 
321  if (type() == ENTER)
322  {
323  Real viol_max = (1 + iterCount) * entertol();
324  int nlinesprinted = 0;
325 
326  for(int i = 0; i < dim(); ++i )
327  {
328  // Minor bound violations happen frequently, so print these
329  // warnings only with verbose level INFO2 and higher.
330  if ((*theFvec)[i] > theUBbound[i] + viol_max) //@ && theUBbound[i] != theLBbound[i])
331  {
332  MSG_INFO2( spxout << "WBOUND01 Invalid upper enter bound " << i
333  << " viol_max: " << viol_max
334  << " Fvec: " << (*theFvec)[i]
335  << " UBbound: "<< theUBbound[i] << std::endl; )
336  nlinesprinted++;
337  }
338  if ((*theFvec)[i] < theLBbound[i] - viol_max) //@ && theUBbound[i] != theLBbound[i])
339  {
340  MSG_INFO2( spxout << "WBOUND02 Invalid lower enter bound " << i
341  << " viol_max: " << viol_max
342  << " Fvec: " << (*theFvec)[i]
343  << " LBbound: "<< theLBbound[i] << std::endl; )
344  nlinesprinted++;
345  }
346  if( nlinesprinted >= 3 )
347  {
348  MSG_INFO2( spxout << "WBOUND10 suppressing further warnings of type WBOUND{01,02} in this round" << std::endl );
349  break;
350  }
351  }
352  }
353  else
354  {
355  Real viol_max = (1 + iterCount) * leavetol();
356  int nlinesprinted = 0;
357 
358  for(int i = 0; i < dim(); ++i )
359  {
360  if ((*theCoPvec)[i] > (*theCoUbound)[i] + viol_max) // && (*theCoUbound)[i] != (*theCoLbound)[i])
361  {
362  MSG_INFO2( spxout << "WBOUND03 Invalid upper cobound " << i
363  << " viol_max: " << viol_max
364  << " CoPvec: " << (*theCoPvec)[i]
365  << " CoUbound: "<< (*theCoUbound)[i] << std::endl; )
366  nlinesprinted++;
367  }
368  if ((*theCoPvec)[i] < (*theCoLbound)[i] - viol_max) // && (*theCoUbound)[i] != (*theCoLbound)[i])
369  {
370  MSG_INFO2( spxout << "WBOUND04 Invalid lower cobound " << i
371  << " viol_max: " << viol_max
372  << " CoPvec: " << (*theCoPvec )[i]
373  << " CoLbound: " << (*theCoLbound)[i]
374  << std::endl; )
375  nlinesprinted++;
376  }
377  if( nlinesprinted >= 3 )
378  {
379  MSG_INFO2( spxout << "WBOUND11 suppressing further warnings of type WBOUND{03,04} in this round" << std::endl );
380  break;
381  }
382  }
383 
384  nlinesprinted = 0;
385  for(int i = 0; i < coDim(); ++i )
386  {
387  if ((*thePvec)[i] > (*theUbound)[i] + viol_max) // && (*theUbound)[i] != (*theLbound)[i])
388  {
389  MSG_INFO2( spxout << "WBOUND05 Invalid upper bound " << i
390  << " viol_max: " << viol_max
391  << " Pvec: " << (*thePvec)[i]
392  << " Ubound: " << (*theUbound)[i] << std::endl; )
393  nlinesprinted++;
394  }
395  if ((*thePvec)[i] < (*theLbound)[i] - viol_max) // && (*theUbound)[i] != (*theLbound)[i])
396  {
397  MSG_INFO2( spxout << "WBOUND06 Invalid lower bound " << i
398  << " viol_max: " << viol_max
399  << " Pvec: " << (*thePvec)[i]
400  << " Lbound: " << (*theLbound)[i] << std::endl; )
401  nlinesprinted++;
402  }
403  if( nlinesprinted >= 3 )
404  {
405  MSG_INFO2( spxout << "WBOUND12 suppressing further warnings of type WBOUND{05,06} in this round" << std::endl );
406  break;
407  }
408  }
409  }
410 }
411 } // namespace soplex
412 
413 //-----------------------------------------------------------------------------
414 //Emacs Local Variables:
415 //Emacs mode:c++
416 //Emacs c-basic-offset:3
417 //Emacs tab-width:8
418 //Emacs indent-tabs-mode:nil
419 //Emacs End:
420 //-----------------------------------------------------------------------------