SoPlex Doxygen Documentation
spxweightst.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 //#define TEST 1
18 
19 #include <assert.h>
20 #include <iostream>
21 
22 #include "spxdefines.h"
23 #include "spxweightst.h"
24 #include "svset.h"
25 #include "sorter.h"
26 
27 namespace soplex
28 {
29 #define EPS 1e-6
30 #define STABLE 1e-3
31 
33 {
34 #ifdef ENABLE_CONSISTENCY_CHECKS
35  return rowWeight.isConsistent()
38  && colUp.isConsistent();
39  // && SPxStarter::isConsistent(); // not yet implemented
40 #else
41  return true;
42 #endif
43 }
44 
45 /* Generating the Starting Basis
46  The generation of a starting basis works as follows: After setting up the
47  preference arrays #weight# and #coWeight#, #Id#s are selected to be dual in
48  a greedy manner. Initially, the first #Id# is taken. Then the next #Id# is
49  checked wheter its vector is linearly dependend of the vectors of the #Id#s
50  allready selected. If not, it is added to them. This is iterated until a
51  full matrix has been constructed.
52 
53  Testing for linear independence is done very much along the lines of LU
54  factorization. A vector is taken, and updated with all previous L-vectors.
55  Then the maximal absolut element is selected as pivot element for computing
56  the L-vector. This is stored for further processing.
57  */
58 
59 /*
60  The following two functions set the status of |id| to primal or dual,
61  respectively.
62  */
64  SPxBasis::Desc& desc,
65  const SPxSolver& base,
66  const SPxId& id)
67 {
68  if (id.isSPxRowId())
69  {
70  int n = base.number(SPxRowId(id));
71 
72  if (base.rhs(n) >= infinity)
73  {
74  if (base.lhs(n) <= -infinity)
76  else
78  }
79  else
80  {
81  if (base.lhs(n) <= -infinity)
83  else if (base.lhs(n) >= base.rhs(n) - base.epsilon())
85  else if (rowRight[n])
87  else
89  }
90  }
91  else
92  {
93  int n = base.number(SPxColId(id));
94  if (base.SPxLP::upper(n) >= infinity)
95  {
96  if (base.SPxLP::lower(n) <= -infinity)
98  else
100  }
101  else
102  {
103  if (base.SPxLP::lower(n) <= -infinity)
105  else if (base.SPxLP::lower(n) >= base.SPxLP::upper(n) - base.epsilon())
107  else if (colUp[n])
109  else
111  }
112  }
113 }
114 
115 // ----------------------------------------------------------------
116 static void setDualStatus(
117  SPxBasis::Desc& desc,
118  const SPxSolver& base,
119  const SPxId& id)
120 {
121  if (id.isSPxRowId())
122  {
123  int n = base.number(SPxRowId(id));
124  desc.rowStatus(n) = base.basis().dualRowStatus(n);
125  }
126  else
127  {
128  int n = base.number(SPxColId(id));
129  desc.colStatus(n) = base.basis().dualColStatus(n);
130  }
131 }
132 // ----------------------------------------------------------------
133 
134 /// Compare class for row weights, used for sorting.
135 struct Compare
136 {
137 public:
138  /// constructor
139  Compare() : weight( 0 ) {}
140 // const SPxSolver* base; ///< the solver
141  const Real* weight; ///< the weights to compare
142 
143  /// compares the weights
144  Real operator()(int i1, int i2) const
145  {
146  return weight[i1] - weight[i2];
147  }
148 };
149 
150 // ----------------------------------------------------------------
151 /**
152  The following method initializes \p pref such that it contains the
153  set of \ref soplex::SPxId "SPxIds" ordered following \p rowWeight and
154  \p colWeight. For the sorting we take the following approach: first
155  we sort the rows, then the columns. Finally we perform a mergesort
156  of both.
157  */
158 static void initPrefs(
159  DataArray<SPxId>& pref,
160  const SPxSolver& base,
161  const DataArray<Real>& rowWeight,
162  const DataArray<Real>& colWeight)
163 {
164  DataArray<int> row(base.nRows());
165  DataArray<int> col(base.nCols());
166  int i;
167  int j;
168  int k;
169 
170  Compare compare;
171 // compare.base = &base;
172 
173  for(i = 0; i < base.nRows(); ++i)
174  row[i] = i;
175 
176  compare.weight = rowWeight.get_const_ptr();
177 
178  sorter_qsort(row.get_ptr(), row.size(), compare); // Sort rows
179 
180  for(i = 0; i < base.nCols(); ++i)
181  col[i] = i;
182 
183  compare.weight = colWeight.get_const_ptr();
184 
185  sorter_qsort(col.get_ptr(), col.size(), compare); // Sort column
186 
187  i = 0;
188  j = 0;
189  k = 0;
190 
191  while(k < pref.size()) // merge sort
192  {
193  if (rowWeight[row[i]] < colWeight[col[j]])
194  {
195  pref[k++] = base.rId(row[i++]);
196 
197  if (i >= base.nRows())
198  while (k < pref.size())
199  pref[k++] = base.cId(col[j++]);
200  }
201  else
202  {
203  pref[k++] = base.cId(col[j++]);
204 
205  if (j >= base.nCols())
206  while (k < pref.size())
207  pref[k++] = base.rId(row[i++]);
208  }
209  }
210  assert(i == base.nRows());
211  assert(j == base.nCols());
212 }
213 
214 // ----------------------------------------------------------------
216 {
217  SPxId tmpId;
218 
219  forbidden.reSize(base.dim());
220  rowWeight.reSize(base.nRows());
221  colWeight.reSize(base.nCols());
222  rowRight.reSize (base.nRows());
223  colUp.reSize (base.nCols());
224 
225  if (base.rep() == SPxSolver::COLUMN)
226  {
227  weight = &colWeight;
228  coWeight = &rowWeight;
229  }
230  else
231  {
232  weight = &rowWeight;
233  coWeight = &colWeight;
234  }
235  assert(weight->size() == base.coDim());
236  assert(coWeight->size() == base.dim());
237 
238  setupWeights(base);
239 
240  SPxBasis::Desc desc(base);
241  // desc.reSize(base.nRows(), base.nCols());
242 
243  DataArray < SPxId > pref(base.nRows() + base.nCols());
244  initPrefs(pref, base, rowWeight, colWeight);
245 
246  int i;
247  int deltai;
248  int j;
249  int sel;
250 
251  for(i = 0; i < forbidden.size(); ++i)
252  forbidden[i] = 0;
253 
254  if (base.rep() == SPxSolver::COLUMN)
255  {
256  i = 0;
257  deltai = 1;
258  }
259  else
260  {
261  i = pref.size() - 1;
262  deltai = -1;
263  }
264 
265  {
266  int dim = base.dim();
267  Real max = 0;
268 
269  for (; i >= 0 && i < pref.size(); i += deltai)
270  {
271  tmpId = pref[i];
272  const SVector& bVec = base.vector(tmpId);
273  sel = -1;
274 
275  // column or row singleton ?
276  if (bVec.size() == 1)
277  {
278  int idx = bVec.index(0);
279 
280  if (forbidden[idx] < 2)
281  {
282  sel = idx;
283  dim += (forbidden[idx] > 0) ? 1 : 0;
284  }
285  }
286  else
287  {
288  max = bVec.maxAbs();
289 
290  int best = base.nRows();
291 
292  for (j = bVec.size(); --j >= 0;)
293  {
294  Real x = bVec.value(j);
295  int k = bVec.index(j);
296  int l = base.coVector(k).size();
297 
298  if (!forbidden[k] && (fabs(x) > STABLE * max) && (l < best))
299  {
300  best = l;
301  sel = k;
302  }
303  }
304  }
305 
306  if (sel >= 0)
307  {
308  MSG_DEBUG(
309  if (pref[i].type() == SPxId::ROW_ID)
310  spxout << "DWEIST01 r" << base.number(pref[i]);
311  else
312  spxout << "DWEIST02 c" << base.number(pref[i]);
313  )
314 
315  forbidden[sel] = 2;
316 
317  if (base.rep() == SPxSolver::COLUMN)
318  setDualStatus(desc, base, pref[i]);
319  else
320  setPrimalStatus(desc, base, pref[i]);
321 
322  for (j = bVec.size(); --j >= 0;)
323  {
324  Real x = bVec.value(j);
325  int k = bVec.index(j);
326 
327  if (!forbidden[k] && (x > EPS * max || -x > EPS * max))
328  {
329  forbidden[k] = 1;
330  --dim;
331  }
332  }
333 
334  if (--dim == 0)
335  {
336  //@ for(++i; i < pref.size(); ++i)
337  if (base.rep() == SPxSolver::COLUMN)
338  {
339  for (i += deltai; i >= 0 && i < pref.size(); i += deltai)
340  setPrimalStatus(desc, base, pref[i]);
341 
342  for (i = forbidden.size(); --i >= 0;)
343  {
344  if (forbidden[i] < 2)
345  setDualStatus(desc, base, base.coId(i));
346  }
347  }
348  else
349  {
350  for (i += deltai; i >= 0 && i < pref.size(); i += deltai)
351  setDualStatus(desc, base, pref[i]);
352 
353  for (i = forbidden.size(); --i >= 0;)
354  {
355  if (forbidden[i] < 2)
356  setPrimalStatus(desc, base, base.coId(i));
357  }
358  }
359  break;
360  }
361  }
362  else if (base.rep() == SPxSolver::COLUMN)
363  setPrimalStatus(desc, base, pref[i]);
364  else
365  setDualStatus(desc, base, pref[i]);
366 #ifndef NDEBUG
367  {
368  int n, m;
369  for (n = 0, m = forbidden.size(); n < forbidden.size(); ++n)
370  m -= (forbidden[n] != 0) ? 1 : 0;
371  assert(m == dim);
372  }
373 #endif // NDEBUG
374  }
375  assert(dim == 0);
376  }
377 
378  base.loadBasis(desc);
379 #ifdef TEST
380  base.init();
381 
382  int changed = 0;
383  const Vector& pvec = base.pVec();
384  for (i = pvec.dim() - 1; i >= 0; --i)
385  {
387  && base.lower(i) > -infinity && pvec[i] > base.maxObj(i))
388  {
389  changed = 1;
391  }
392  else if (desc.colStatus(i) == SPxBasis::Desc::P_ON_LOWER
393  && base.upper(i) < infinity && pvec[i] < base.maxObj(i))
394  {
395  changed = 1;
397  }
398  }
399 
400  if (changed)
401  {
402  std::cout << "changed basis\n";
403  base.loadBasis(desc);
404  }
405  else
406  std::cout << "nothing changed\n";
407 #endif // TEST
408 }
409 
410 // ----------------------------------------------------------------
411 
412 /* Computation of Weights
413  */
415 {
416  const SPxSolver& base = bse;
417  const Vector& obj = bse.maxObj();
418  const Vector& low = bse.SPxLP::lower();
419  const Vector& up = bse.SPxLP::upper();
420  const Vector& rhs = bse.rhs();
421  const Vector& lhs = bse.lhs();
422  int i;
423 
424  Real maxabs = 1.0;
425 
426  // find absolut biggest entry in bounds and left-/right hand side
427  for (i = 0; i < bse.nCols(); i++)
428  {
429  if ((up[i] < infinity) && (fabs(up[i]) > maxabs))
430  maxabs = fabs(up[i]);
431 
432  if ((low[i] > -infinity) && (fabs(low[i]) > maxabs))
433  maxabs = fabs(low[i]);
434  }
435  for (i = 0; i < bse.nRows(); i++)
436  {
437  if ((rhs[i] < infinity) && (fabs(rhs[i]) > maxabs))
438  maxabs = fabs(rhs[i]);
439 
440  if ((lhs[i] > -infinity) && (fabs(lhs[i]) > maxabs))
441  maxabs = fabs(lhs[i]);
442  }
443 
444  /**@todo The comments are wrong. The first is for dual simplex and
445  * the secound for primal one. Is anything else wrong?
446  * Also the values are nearly the same for both cases.
447  * Should this be ? Changed the values for
448  * r_fixed to 0 because of maros-r7. It is not clear why
449  * this makes a difference because all constraints in that
450  * instance are of equality type.
451  * Why is rowRight sometimes not set?
452  */
453  if (bse.rep() * bse.type() > 0) // primal simplex
454  {
455  const Real ax = 1e-3 / obj.maxAbs();
456  const Real bx = 1.0 / maxabs;
457  const Real nne = ax / lhs.dim(); // 1e-4 * ax;
458  const Real c_fixed = 1e+5;
459  const Real r_fixed = 0; // TK20010103: was 1e+4 (maros-r7)
460  const Real c_dbl_bounded = 1e+1;
461  const Real r_dbl_bounded = 0;
462  const Real c_bounded = 1e+1;
463  const Real r_bounded = 0;
464  const Real c_free = -1e+4;
465  const Real r_free = -1e+5;
466 
467  for (i = bse.nCols() - 1; i >= 0; i--)
468  {
469  Real n = nne * (bse.colVector(i).size() - 1);
470  Real x = ax * obj[i];
471  Real u = bx * up [i];
472  Real l = bx * low[i];
473 
474  if (up[i] < infinity)
475  {
476  if (fabs(low[i] - up[i]) < base.epsilon())
477  colWeight[i] = c_fixed + n + fabs(x);
478  else if (low[i] > -infinity)
479  {
480  colWeight[i] = c_dbl_bounded + l - u + n;
481 
482  l = fabs(l);
483  u = fabs(u);
484 
485  if (u < l)
486  {
487  colUp[i] = true;
488  colWeight[i] += x;
489  }
490  else
491  {
492  colUp[i] = false;
493  colWeight[i] -= x;
494  }
495  }
496  else
497  {
498  colWeight[i] = c_bounded - u + x + n;
499  colUp[i] = true;
500  }
501  }
502  else
503  {
504  if (low[i] > -infinity)
505  {
506  colWeight[i] = c_bounded + l + n - x;
507  colUp[i] = false;
508  }
509  else
510  {
511  colWeight[i] = c_free + n - fabs(x);
512  }
513  }
514  }
515 
516  for (i = bse.nRows() - 1; i >= 0; i--)
517  {
518  if (rhs[i] < infinity)
519  {
520  if (fabs(lhs[i] - rhs[i]) < base.epsilon())
521  {
522  rowWeight[i] = r_fixed;
523  }
524  else if (lhs[i] > -infinity)
525  {
526  Real u = bx * rhs[i];
527  Real l = bx * lhs[i];
528 
529  rowWeight[i] = r_dbl_bounded + l - u;
530  rowRight[i] = fabs(u) < fabs(l);
531  }
532  else
533  {
534  rowWeight[i] = r_bounded - bx * rhs[i];
535  rowRight[i] = true;
536  }
537  }
538  else
539  {
540  if (lhs[i] > -infinity)
541  {
542  rowWeight[i] = r_bounded + bx * lhs[i];
543  rowRight[i] = false;
544  }
545  else
546  {
547  rowWeight[i] = r_free;
548  }
549  }
550  }
551  }
552  else
553  {
554  assert(bse.rep() * bse.type() < 0); // dual simplex
555 
556  const Real ax = 1.0 / obj.maxAbs();
557  const Real bx = 1e-2 / maxabs;
558  const Real nne = 1e-4 * bx;
559  const Real c_fixed = 1e+5;
560  const Real r_fixed = 1e+4;
561  const Real c_dbl_bounded = 1;
562  const Real r_dbl_bounded = 0;
563  const Real c_bounded = 0;
564  const Real r_bounded = 0;
565  const Real c_free = -1e+4;
566  const Real r_free = -1e+5;
567 
568  for (i = bse.nCols() - 1; i >= 0; i--)
569  {
570  Real n = nne * (bse.colVector(i).size() - 1);
571  Real x = ax * obj[i];
572  Real u = bx * up [i];
573  Real l = bx * low[i];
574 
575  if (up[i] < infinity)
576  {
577  if (fabs(low[i] - up[i]) < base.epsilon())
578  colWeight[i] = c_fixed + n + fabs(x);
579  else if (low[i] > -infinity)
580  {
581  if (x > 0)
582  {
583  colWeight[i] = c_dbl_bounded + x - u + n;
584  colUp[i] = true;
585  }
586  else
587  {
588  colWeight[i] = c_dbl_bounded - x + l + n;
589  colUp[i] = false;
590  }
591  }
592  else
593  {
594  colWeight[i] = c_bounded - u + x + n;
595  colUp[i] = true;
596  }
597  }
598  else
599  {
600  if (low[i] > -infinity)
601  {
602  colWeight[i] = c_bounded - x + l + n;
603  colUp[i] = false;
604  }
605  else
606  colWeight[i] = c_free + n - fabs(x);
607  }
608  }
609 
610  for (i = bse.nRows() - 1; i >= 0; i--)
611  {
612  const Real len1 = 1; // (bse.rowVector(i).length() + bse.epsilon());
613  Real n = 0; // nne * (bse.rowVector(i).size() - 1);
614  Real u = bx * len1 * rhs[i];
615  Real l = bx * len1 * lhs[i];
616  Real x = ax * len1 * (obj * bse.rowVector(i));
617 
618  if (rhs[i] < infinity)
619  {
620  if (fabs(lhs[i] - rhs[i]) < base.epsilon())
621  rowWeight[i] = r_fixed + n + fabs(x);
622  else if (lhs[i] > -infinity)
623  {
624  if (x > 0)
625  {
626  rowWeight[i] = r_dbl_bounded + x - u + n;
627  rowRight[i] = true;
628  }
629  else
630  {
631  rowWeight[i] = r_dbl_bounded - x + l + n;
632  rowRight[i] = false;
633  }
634  }
635  else
636  {
637  rowWeight[i] = r_bounded - u + n + x;
638  rowRight[i] = true;
639  }
640  }
641  else
642  {
643  if (lhs[i] > -infinity)
644  {
645  rowWeight[i] = r_bounded + l + n - x;
646  rowRight[i] = false;
647  }
648  else
649  {
650  rowWeight[i] = r_free + n - fabs(x);
651  }
652  }
653  }
654  }
655 
656  MSG_DEBUG({
657  for(i = 0; i < bse.nCols(); i++)
658  spxout << "C i= " << i
659  << " up= " << colUp[i]
660  << " w= " << colWeight[i]
661  << std::endl;
662  for(i = 0; i < bse.nRows(); i++)
663  spxout << "R i= " << i
664  << " rr= " << rowRight[i]
665  << " w= " << rowWeight[i]
666  << std::endl;
667  })
668 }
669 } // namespace soplex
670 
671 //-----------------------------------------------------------------------------
672 //Emacs Local Variables:
673 //Emacs mode:c++
674 //Emacs c-basic-offset:3
675 //Emacs tab-width:8
676 //Emacs indent-tabs-mode:nil
677 //Emacs End:
678 //-----------------------------------------------------------------------------