SoPlex Doxygen Documentation
spxdevexpr.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 #include "spxdefines.h"
17 #include "spxdevexpr.h"
18 
19 #define DEVEX_REFINETOL 2.0
20 
21 namespace soplex
22 {
23 
25 {
26  thesolver = base;
27  setRep(base->rep());
28  assert(isConsistent());
29 }
30 
32 {
33 #ifdef ENABLE_CONSISTENCY_CHECKS
34  if (thesolver != 0)
35  if (penalty.dim() != thesolver->coDim()
36  || coPenalty.dim() != thesolver->dim())
37  return MSGinconsistent("SPxDevexPR");
38 #endif
39 
40  return true;
41 }
42 
44 {
45  int i;
46  if (tp == SPxSolver::ENTER)
47  {
48  for (i = penalty.dim(); --i >= 0;)
49  penalty[i] = 2;
50  for (i = coPenalty.dim(); --i >= 0;)
51  coPenalty[i] = 2;
52  }
53  else
54  {
55  for (i = coPenalty.dim(); --i >= 0;)
56  coPenalty[i] = 1;
57  }
58  assert(isConsistent());
59 }
60 
62 {
63  init(tp);
64  refined = false;
65 }
66 
67 /**@todo suspicious: Shouldn't the relation between dim, coDim, Vecs,
68  * and CoVecs be influenced by the representation ?
69  */
71 {
72  if (thesolver != 0)
73  {
76  assert(isConsistent());
77  }
78 }
79 
81 {
82  int retid;
83  Real val;
84 
85 #ifdef PARTIAL_PRICING
86  retid = selectLeavePart(val, theeps);
87 #else
89  retid = selectLeaveSparse(val, theeps);
90  else
91  retid = selectLeaveX(val, theeps);
92 #endif
93  if ( retid < 0 && !refined )
94  {
95  refined = true;
96  MSG_INFO3( spxout << "WDEVEX02 trying refinement step..\n"; )
97  retid = selectLeaveX(val, theeps/DEVEX_REFINETOL);
98  }
99 
100  return retid;
101 }
102 
103 int SPxDevexPR::selectLeaveX(Real& best, Real feastol, int start, int incr)
104 {
105  Real x;
106 
107  const Real* fTest = thesolver->fTest().get_const_ptr();
108  const Real* cpen = coPenalty.get_const_ptr();
109  Real bstX = 0;
110  int bstI = -1;
111  int end = coPenalty.dim();
112 
113  for (; start < end; start += incr)
114  {
115  if (fTest[start] < -feastol)
116  {
117  x = fTest[start] * fTest[start] / cpen[start];
118  if (x > bstX)
119  {
120  bstX = x;
121  bstI = start;
122  last = cpen[start];
123  }
124  }
125  }
126  best = bstX;
127  return bstI;
128 }
129 
131 {
132  Real x;
133 
134  const Real* fTest = thesolver->fTest().get_const_ptr();
135  const Real* cpen = coPenalty.get_const_ptr();
136  Real bstX = 0;
137  int bstI = -1;
138  int dim = coPenalty.dim();
139  int count = 0;
140  int oldstartpricing = startpricing;
141  int end = oldstartpricing + IMPROVEMENT_STEPLENGTH;
142 
143  for (int i = oldstartpricing; i < dim; ++i)
144  {
145  Real fTesti = fTest[i];
146  if (fTesti < -feastol)
147  {
148  Real cpeni = cpen[i];
149  x = fTesti * fTesti / cpeni;
150  if (x > bstX * IMPROVEMENT_THRESHOLD)
151  {
152  bstX = x;
153  bstI = i;
154  last = cpeni;
155  end = i + IMPROVEMENT_STEPLENGTH;
156 // if (count == 0)
157  startpricing = (i + 1) % dim;
158  ++count;
159  }
160  }
161  if (i >= end && count >= MAX_PRICING_CANDIDATES)
162  break;
163  }
164 
165  if (end < dim && count >= MAX_PRICING_CANDIDATES)
166  {
167  best = bstX;
168  return bstI;
169  }
170  else
171  {
172  end -= dim;
173  }
174 
175  for (int i = 0; i < oldstartpricing; ++i)
176  {
177  Real fTesti = fTest[i];
178  if (fTesti < -feastol)
179  {
180  Real cpeni = cpen[i];
181  x = fTesti * fTesti / cpeni;
182  if (x > bstX * IMPROVEMENT_THRESHOLD)
183  {
184  bstX = x;
185  bstI = i;
186  last = cpeni;
187  end = i + IMPROVEMENT_STEPLENGTH;
188 // if (count == 0)
189  startpricing = (i + 1) % dim;
190  ++count;
191  }
192  }
193  if (i >= end && count >= MAX_PRICING_CANDIDATES)
194  break;
195  }
196 
197  best = bstX;
198  return bstI;
199 }
200 
202 {
203  Real x;
204 
205  const Real* fTest = thesolver->fTest().get_const_ptr();
206  const Real* cpen = coPenalty.get_const_ptr();
207  Real bstX = 0;
208  int bstI = -1;
209  int idx = -1;
210  Real fTesti;
211  Real coPeni;
212 
213  for (int i = thesolver->infeasibilities.size() - 1; i >= 0; --i)
214  {
215  idx = thesolver->infeasibilities.index(i);
216  fTesti = fTest[idx];
217  if (fTesti < -feastol)
218  {
219  coPeni = cpen[idx];
220  x = fTesti * fTesti / coPeni;
221  if (x > bstX)
222  {
223  bstX = x;
224  bstI = idx;
225  last = coPeni;
226  }
227  }
228  else
229  {
231 
232  assert(thesolver->isInfeasible[idx]);
233  thesolver->isInfeasible[idx] = false;
234  }
235  }
236  best = bstX;
237  return bstI;
238 }
239 
240 
241 void SPxDevexPR::left4(int n, SPxId id)
242 {
243  left4X(n, id, 0, 1);
244 }
245 
246 void SPxDevexPR::left4X(int n, const SPxId& id, int start, int incr)
247 {
248  if (id.isValid())
249  {
250  int i, j;
251  Real x;
252  const Real* rhoVec = thesolver->fVec().delta().values();
253  Real rhov_1 = 1 / rhoVec[n];
254  Real beta_q = thesolver->coPvec().delta().length2() * rhov_1 * rhov_1;
255 
256 #ifndef NDEBUG
257  if (fabs(rhoVec[n]) < theeps)
258  {
259  MSG_ERROR( spxout << "WDEVEX01: rhoVec = "
260  << rhoVec[n] << " with smaller absolute value than theeps = " << theeps << std::endl; )
261  }
262 #endif // NDEBUG
263 
264  // Update #coPenalty# vector
265  const IdxSet& rhoIdx = thesolver->fVec().idx();
266  int len = thesolver->fVec().idx().size();
267  for (i = len - 1 - start; i >= 0; i -= incr)
268  {
269  j = rhoIdx.index(i);
270  x = rhoVec[j] * rhoVec[j] * beta_q;
271  // if(x > coPenalty[j])
272  coPenalty[j] += x;
273  }
274 
275  coPenalty[n] = beta_q;
276  }
277 }
278 
279 
280 
282 {
283  assert(thesolver != 0);
284 
285  SPxId enterId;
286 
287  enterId = selectEnterX(theeps);
288 
289  if( !enterId.isValid() && !refined )
290  {
291  refined = true;
292  MSG_INFO3( spxout << "WDEVEX02 trying refinement step..\n"; )
294  }
295 
296  return enterId;
297 }
298 
299 // choose the best entering index among columns and rows but prefer sparsity
301 {
302  SPxId enterId;
303  SPxId enterIdCo;
304  Real best;
305  Real bestCo;
306 
307  best = 0;
308  bestCo = 0;
309  enterId = (thesolver->sparsePricingEnter) ? selectEnterSparseDim(best, tol) : selectEnterDenseDim(best, tol);
310  enterIdCo = (thesolver->sparsePricingEnterCo) ? selectEnterSparseCoDim(bestCo, tol) : selectEnterDenseCoDim(bestCo, tol);
311 
312  // prefer slack indices to reduce nonzeros in basis matrix
313  if( enterId.isValid() && (best > SPARSITY_TRADEOFF * bestCo || !enterIdCo.isValid()) )
314  return enterId;
315  else
316  return enterIdCo;
317 }
318 
320 {
321  const Real* cTest = thesolver->coTest().get_const_ptr();
322  const Real* cpen = coPenalty.get_const_ptr();
323  int end = coPenalty.dim();
324  int enterIdx = -1;
325  int idx;
326  Real coTesti;
327  Real coPeni;
328  Real x;
329 
330  assert(end == thesolver->coTest().dim());
331  for(int i = thesolver->infeasibilities.size() -1; i >= 0; --i)
332  {
333  idx = thesolver->infeasibilities.index(i);
334  coTesti = cTest[idx];
335  if (coTesti < -feastol)
336  {
337  coPeni = cpen[idx];
338  x = coTesti * coTesti / coPeni;
339  if (x > best)
340  {
341  best = x;
342  enterIdx = idx;
343  last = cpen[idx];
344  }
345  }
346  else
347  {
349  assert(thesolver->isInfeasible[idx]);
350  thesolver->isInfeasible[idx] = false;
351  }
352  }
353  if (enterIdx >= 0)
354  return thesolver->coId(enterIdx);
355 
356  SPxId none;
357  return none;
358 }
359 
360 
362 {
363  const Real* test = thesolver->test().get_const_ptr();
364  const Real* pen = penalty.get_const_ptr();
365  int end = penalty.dim();
366  int enterIdx = -1;
367  int idx;
368  Real testi;
369  Real peni;
370  Real x;
371 
372  assert(end == thesolver->test().dim());
373  for (int i = thesolver->infeasibilitiesCo.size() -1; i >= 0; --i)
374  {
376  testi = test[idx];
377  if (testi < -feastol)
378  {
379  peni = pen[idx];
380  x = testi * testi / peni;
381  if (x > best)
382  {
383  best = x;
384  enterIdx = idx;
385  last = pen[idx];
386  }
387  }
388  else
389  {
391  assert(thesolver->isInfeasibleCo[idx]);
392  thesolver->isInfeasibleCo[idx] = false;
393  }
394  }
395 
396  if (enterIdx >= 0)
397  return thesolver->id(enterIdx);
398 
399  SPxId none;
400  return none;
401 }
402 
403 
404 SPxId SPxDevexPR::selectEnterDenseDim(Real& best, Real feastol, int start, int incr)
405 {
406  const Real* cTest = thesolver->coTest().get_const_ptr();
407  const Real* cpen = coPenalty.get_const_ptr();
408  int end = coPenalty.dim();
409  int enterIdx = -1;
410  Real x;
411 
412  assert(end == thesolver->coTest().dim());
413  for (; start < end; start += incr)
414  {
415  if (cTest[start] < -feastol)
416  {
417  x = cTest[start] * cTest[start] / cpen[start];
418  if (x > best)
419  {
420  best = x;
421  enterIdx = start;
422  last = cpen[start];
423  }
424  }
425  }
426 
427  if (enterIdx >= 0)
428  return thesolver->coId(enterIdx);
429 
430  SPxId none;
431  return none;
432 }
433 
434 
435 SPxId SPxDevexPR::selectEnterDenseCoDim(Real& best, Real feastol, int start, int incr)
436 {
437  const Real* test = thesolver->test().get_const_ptr();
438  const Real* pen = penalty.get_const_ptr();
439  int end = penalty.dim();
440  int enterIdx = -1;
441  Real x;
442 
443  assert(end == thesolver->test().dim());
444  for (; start < end; start += incr)
445  {
446  if (test[start] < -feastol)
447  {
448  x = test[start] * test[start] / pen[start];
449  if (x > best)
450  {
451  best = x;
452  enterIdx = start;
453  last = pen[start];
454  }
455  }
456  }
457 
458  if (enterIdx >= 0)
459  return thesolver->id(enterIdx);
460 
461  SPxId none;
462  return none;
463 }
464 
465 
467 {
468  entered4X(id, n, 0, 1, 0, 1);
469 }
470 
471 /**@todo suspicious: the pricer should be informed, that variable id
472  has entered the basis at position n, but the id is not used here
473  (this is true for all pricers)
474 */
475 void SPxDevexPR::entered4X(SPxId /*id*/, int n,
476  int start1, int incr1, int start2, int incr2)
477 {
478  if (n >= 0 && n < thesolver->dim())
479  {
480  const Real* pVec = thesolver->pVec().delta().values();
481  const IdxSet& pIdx = thesolver->pVec().idx();
482  const Real* coPvec = thesolver->coPvec().delta().values();
483  const IdxSet& coPidx = thesolver->coPvec().idx();
484  Real xi_p = 1 / thesolver->fVec().delta()[n];
485  int i, j;
486 
487  assert(thesolver->fVec().delta()[n] > thesolver->epsilon()
488  || thesolver->fVec().delta()[n] < -thesolver->epsilon());
489 
490  xi_p = xi_p * xi_p * last;
491 
492  for (j = coPidx.size() - 1 - start1; j >= 0; j -= incr1)
493  {
494  i = coPidx.index(j);
495  coPenalty[i] += xi_p * coPvec[i] * coPvec[i];
496  if (coPenalty[i] <= 1 || coPenalty[i] > 1e+6)
497  {
499  return;
500  }
501  }
502 
503  for (j = pIdx.size() - 1 - start2; j >= 0; j -= incr2)
504  {
505  i = pIdx.index(j);
506  penalty[i] += xi_p * pVec[i] * pVec[i];
507  if (penalty[i] <= 1 || penalty[i] > 1e+6)
508  {
510  return;
511  }
512  }
513  }
514 }
515 
517 {
518  int initval = (thesolver->type() == SPxSolver::ENTER) ? 2 : 1;
519  n = penalty.dim();
521  for (int i = penalty.dim()-1; i >= n; --i )
522  penalty[i] = initval;
523 }
524 
526 {
527  int initval = (thesolver->type() == SPxSolver::ENTER) ? 2 : 1;
528  n = coPenalty.dim();
530  for (int i = coPenalty.dim()-1; i >= n; --i)
531  coPenalty[i] = initval;
532 }
533 
534 } // namespace soplex
535 
536 //-----------------------------------------------------------------------------
537 //Emacs Local Variables:
538 //Emacs mode:c++
539 //Emacs c-basic-offset:3
540 //Emacs tab-width:8
541 //Emacs indent-tabs-mode:nil
542 //Emacs End:
543 //-----------------------------------------------------------------------------