SoPlex Doxygen Documentation
ssvector.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 <iostream>
17 #include <iomanip>
18 #include <assert.h>
19 
20 #include "spxdefines.h"
21 #include "ssvector.h"
22 #include "svset.h"
23 #include "exceptions.h"
24 
25 /**@file ssvector.cpp
26  * @todo There is a lot pointer arithmetic done here. It is not clear if
27  * this is an advantage at all. See all the function int() casts.
28  * @todo Several operations like maxAbs could setup the vector while
29  * computing the result.
30  */
31 namespace soplex
32 {
33 #define MARKER 1e-100
34 
35 static const Real shortProductFactor = 0.5;
36 
37 void SSVector::setMax(int newmax)
38 {
39  assert(idx != 0);
40  assert(newmax != 0);
41  assert(newmax >= IdxSet::size());
42 
43  // len = (newmax < IdxSet::max()) ? IdxSet::max() : newmax;
44  len = newmax;
45 
47 }
48 
49 void SSVector::reDim (int newdim)
50 {
51  for (int i = IdxSet::size() - 1; i >= 0; --i)
52  if (index(i) >= newdim)
53  remove(i);
54  DVector::reDim(newdim);
55  setMax(DVector::memSize() + 1);
56  assert(isConsistent());
57 }
58 
59 void SSVector::reMem(int newsize)
60 {
61  DVector::reSize(newsize);
62  assert(isConsistent());
63  setMax(DVector::memSize() + 1);
64 }
65 
67 {
68  if (isSetup())
69  {
70  for(int i = 0; i < num; ++i)
71  val[idx[i]] = 0.0;
72  }
73  else
74  Vector::clear();
75 
76  IdxSet::clear();
77  setupStatus = true;
78  assert(isConsistent());
79 }
80 
81 void SSVector::setValue(int i, Real x)
82 {
83  assert(i >= 0 && i < DVector::dim());
84 
85  if (isSetup())
86  {
87  int n = number(i);
88 
89  if (n < 0)
90  {
91  if (isNotZero(x, epsilon))
92  IdxSet::add(1, &i);
93  }
94  else if (x == 0)
95  clearNum(n);
96  }
97  val[i] = x;
98 
99  assert(isConsistent());
100 }
101 
102 // #ifdef USE_OLD // old version
103 // void SSVector::setup()
104 // {
105 // if (!isSetup())
106 // {
107 // IdxSet::clear();
108 //
109 // // #define TWO_LOOPS
110 // #ifdef TWO_LOOPS
111 // int i = 0;
112 // int n = 0;
113 // int* id = idx;
114 // Real* v = val;
115 // const Real* end = val + dim();
116 //
117 // while (v < end)
118 // {
119 // id[n] = i++;
120 // n += (*v++ != 0);
121 // }
122 //
123 // Real x;
124 // int* ii = idx;
125 // int* last = idx + n;
126 // v = val;
127 //
128 // for (; id < last; ++id)
129 // {
130 // x = v[*id];
131 // if (isNotZero(x, epsilon))
132 // *ii++ = *id;
133 // else
134 // v[*id] = 0;
135 // }
136 // num = ii - idx;
137 //
138 // #else
139 //
140 // if (dim() <= 1)
141 // {
142 // if (dim())
143 // {
144 // if (isNotZero(*val, epsilon))
145 // IdxSet::add(0);
146 // else
147 // *val = 0;
148 // }
149 // }
150 // else
151 // {
152 // int* ii = idx;
153 // Real* v = val;
154 // Real* end = v + dim() - 1;
155 //
156 // /* setze weissen Elefanten */
157 // Real last = *end;
158 // *end = MARKER;
159 //
160 // /* erstes element extra */
161 // if (isNotZero(*v, epsilon))
162 // *ii++ = 0;
163 // else
164 // *v = 0.0;
165 //
166 // for(;;)
167 // {
168 // while (*++v == 0.0)
169 // ;
170 // if (isNotZero(*v, epsilon))
171 // {
172 // *ii++ = int(v - val);
173 // }
174 // else
175 // {
176 // *v = 0.0;
177 // if (v == end)
178 // break;
179 // }
180 // }
181 // /* fange weissen Elefanten wieder ein */
182 // if (isNotZero(last, epsilon))
183 // {
184 // *v = last;
185 // *ii++ = dim() - 1;
186 // }
187 // else
188 // *v = 0;
189 //
190 // num = int(ii - idx);
191 // }
192 //
193 // #endif
194 //
195 // setupStatus = true;
196 // assert(isConsistent());
197 // }
198 // }
199 // #else // new version, not yet fully tested
201 {
202  if (!isSetup())
203  {
204  IdxSet::clear();
205 
206  num = 0;
207 
208  for(int i = 0; i < dim(); ++i)
209  {
210  if (val[i] != 0.0)
211  {
212  if (isZero(val[i], epsilon))
213  val[i] = 0.0;
214  else
215  {
216  idx[num] = i;
217  num++;
218  }
219  }
220  }
221  setupStatus = true;
222  assert(isConsistent());
223  }
224 }
225 // #endif
226 
228 {
229  Vector::operator+=(vec);
230 
231  if (isSetup())
232  {
233  setupStatus = false;
234  setup();
235  }
236  return *this;
237 }
238 
240 {
241  Vector::operator+=(vec);
242 
243  if (isSetup())
244  {
245  setupStatus = false;
246  setup();
247  }
248  return *this;
249 }
250 
252 {
253  for (int i = 0; i < vec.size(); ++i)
254  val[vec.index(i)] += vec.value(i);
255 
256  if (isSetup())
257  {
258  setupStatus = false;
259  setup();
260  }
261  return *this;
262 }
263 
265 {
266  Vector::operator-=(vec);
267 
268  if (isSetup())
269  {
270  setupStatus = false;
271  setup();
272  }
273  return *this;
274 }
275 
277 {
278  Vector::operator-=(vec);
279 
280  if (isSetup())
281  {
282  setupStatus = false;
283  setup();
284  }
285  return *this;
286 }
287 
289 {
290  if (vec.isSetup())
291  {
292  for (int i = 0; i < vec.size(); ++i)
293  val[vec.index(i)] -= vec.value(i);
294  }
295  else
296  {
298  }
299 
300  if (isSetup())
301  {
302  setupStatus = false;
303  setup();
304  }
305  return *this;
306 }
307 
309 {
310  assert(isSetup());
311 
312  for (int i = 0; i < size(); ++i)
313  val[index(i)] *= x;
314 
315  assert(isConsistent());
316 
317  return *this;
318 }
319 
321 {
322  if (isSetup())
323  {
324  Real maxabs = REAL(0.0);
325 
326  for(int i = 0; i < num; ++i)
327  {
328  Real x = fabs(val[idx[i]]);
329 
330  if (x > maxabs)
331  maxabs = x;
332  }
333  return maxabs;
334  }
335  else
336  return Vector::maxAbs();
337 }
338 
340 {
341  Real x = REAL(0.0);
342 
343  if (isSetup())
344  {
345  for(int i = 0; i < num; ++i)
346  x += val[idx[i]] * val[idx[i]];
347  }
348  else
349  x = Vector::length2();
350 
351  return x;
352 }
353 
355 {
356  return sqrt(length2());
357 }
358 
359 #if 0 // buggy and not used
360 /* @todo check if really not used or if the Vector version is used instead.
361  */
362 SSVector& SSVector::multAdd(Real xx, const SSVector& svec)
363 {
364  if (svec.isSetup())
365  {
366  if (isSetup())
367  {
368  int i, j;
369  Real x;
370 
371  for (i = svec.size() - 1; i >= 0; --i)
372  {
373  j = svec.index(i);
374  if (val[j])
375  {
376  x = val[j] + xx * svec.value(i);
377  if (isNotZero(x, epsilon))
378  val[j] = x;
379  else
380  {
381  val[j] = 0;
382  for (--i; i >= 0; --i)
383  val[svec.index(i)] += xx * svec.value(i);
384  unSetup();
385  break;
386  }
387  }
388  else
389  {
390  x = xx * svec.value(i);
391  if (isNotZero(x, epsilon))
392  {
393  val[j] = x;
394  addIdx(j);
395  }
396  }
397  }
398  }
399  else
400  Vector::multAdd(xx, svec);
401  }
402  else
403  {
404  /**@todo this code does not work, because in is never something
405  * added to v. Also the idx will not be setup correctly
406  * Fortunately the whole function seems not to be called
407  * at all.
408  */
409  throw SPxInternalCodeException("XSSVEC01 This should never happen.");
410 
411  Real y;
412  int* ii = idx;
413  Real* v = val;
414  Real* rv = static_cast<Real*>(svec.val);
415  Real* last = rv + svec.dim() - 1;
416  Real x = *last;
417 
418  *last = MARKER;
419  for(;;)
420  {
421  while (!*rv)
422  {
423  ++rv;
424  ++v;
425  }
426  y = *rv++ * xx;
427  if (isNotZero(y, epsilon))
428  {
429  *ii++ = int(v - val);
430  *v++ = y;
431  }
432  else if (rv == last)
433  break;
434  else
435  v++;
436  }
437  *rv = x;
438 
439  x *= xx;
440  if (isNotZero(x, epsilon))
441  {
442  *ii++ = int(v - val);
443  *v = x;
444  }
445  num = int(ii - idx);
446 
447  setupStatus = true;
448  }
449 
450  assert(isConsistent());
451  return *this;
452 }
453 #endif // 0
454 
455 ///@todo SSVector::multAdd() should be rewritten without pointer arithmetic.
457 {
458  if (isSetup())
459  {
460  int i, j;
461  Real x;
462  Real* v = val;
463  int adjust = 0;
464 
465  for (i = svec.size() - 1; i >= 0; --i)
466  {
467  j = svec.index(i);
468  if (v[j])
469  {
470  x = v[j] + xx * svec.value(i);
471  if (isNotZero(x, epsilon))
472  v[j] = x;
473  else
474  {
475  adjust = 1;
476  v[j] = MARKER;
477  }
478  }
479  else
480  {
481  x = xx * svec.value(i);
482  if (isNotZero(x, epsilon))
483  {
484  v[j] = x;
485  addIdx(j);
486  }
487  }
488  }
489 
490  if (adjust)
491  {
492  int* iptr = idx;
493  int* iiptr = idx;
494  int* endptr = idx + num;
495  for (; iptr < endptr; ++iptr)
496  {
497  x = v[*iptr];
498  if (isNotZero(x, epsilon))
499  *iiptr++ = *iptr;
500  else
501  v[*iptr] = 0;
502  }
503  num = int(iiptr - idx);
504  }
505  }
506  else
507  Vector::multAdd(xx, svec);
508 
509  assert(isConsistent());
510  return *this;
511 }
512 
514 {
515  Vector::multAdd(x, vec);
516 
517  if (isSetup())
518  {
519  setupStatus = false;
520  setup();
521  }
522  return *this;
523 }
524 
526 {
527  assert(rhs.isConsistent());
528 
529  if (this != &rhs)
530  {
531  clear();
532  epsilon = rhs.epsilon;
533  setMax(rhs.max());
534  DVector::reDim(rhs.dim());
535 
536  if (rhs.isSetup())
537  {
538  IdxSet::operator=(rhs);
539 
540  for(int i = 0; i < size(); ++i)
541  {
542  int j = index(i);
543  val[j] = rhs.val[j];
544  }
545  }
546  else
547  {
548  num = 0;
549 
550  for(int i = 0; i < rhs.dim(); ++i)
551  {
552  if (isNotZero(rhs.val[i], epsilon))
553  {
554  val[i] = rhs.val[i];
555  idx[num] = i;
556  num++;
557  }
558  }
559  }
560  setupStatus = true;
561  }
562  assert(isConsistent());
563 
564  return *this;
565 }
566 
567 // setup rhs and assign to this
569 {
570  clear();
571  epsilon = rhs.epsilon;
572  setMax(rhs.max());
573  DVector::reDim(rhs.dim());
574 
575  if (rhs.isSetup())
576  {
577  IdxSet::operator=(rhs);
578 
579  for(int i = 0; i < size(); ++i)
580  {
581  int j = index(i);
582  val[j] = rhs.val[j];
583  }
584  }
585  else
586  {
587  num = 0;
588 
589  for(int i = 0; i < rhs.dim(); ++i)
590  {
591  if (rhs.val[i] != 0.0)
592  {
593  if (isNotZero(rhs.val[i], epsilon))
594  {
595  rhs.idx[num] = i;
596  idx[num] = i;
597  val[i] = rhs.val[i];
598  num++;
599  }
600  else
601  {
602  rhs.val[i] = 0.0;
603  }
604  }
605  }
606  rhs.num = num;
607  rhs.setupStatus = true;
608  }
609  setupStatus = true;
610 
611  assert(rhs.isConsistent());
612  assert(isConsistent());
613 }
614 
615 
617 {
618  clear();
619  return assign(rhs);
620 }
621 
622 // @todo implementation of SSVector::assign() could be put into operator=()
624 {
625  assert(rhs.dim() <= Vector::dim());
626 
627  num = 0;
628 
629  for(int i = 0; i < rhs.size(); ++i)
630  {
631  int k = rhs.index(i);
632  Real v = rhs.value(i);
633 
634  if (isZero(v, epsilon))
635  val[k] = REAL(0.0);
636  else
637  {
638  val[k] = v;
639  idx[num++] = k;
640  }
641  }
642  setupStatus = true;
643 
644  assert(isConsistent());
645 
646  return *this;
647 }
648 
650 {
651  assert(x.isSetup());
652  assert(x.size() == 1);
653 
654  // get the nonzero value of x and the corresponding vector in A:
655  const int nzidx = x.idx[0];
656  const Real nzval = x.val[nzidx];
657  const SVector& Ai = A [nzidx];
658 
659  // compute A[nzidx] * nzval:
660  if ( isZero(nzval, epsilon) || Ai.size() == 0 )
661  clear(); // this := zero vector
662  else
663  {
664  num = Ai.size();
665  for ( register int j = 0; j < num; j++ )
666  {
667  const SVector::Element& Aij = Ai.element(j);
668  idx[j] = Aij.idx;
669  val[Aij.idx] = nzval * Aij.val;
670  }
671  }
672 
673  assert(isConsistent());
674  return *this;
675 }
676 
678 {
679  assert(x.isSetup());
680 
681  if (x.size() == 0) // x can be setup but have size 0 => this := zero vector
682  {
683  clear();
684  return *this;
685  }
686 
687  // compute x[0] * A[0]
688  int curidx = x.idx[0];
689  const Real x0 = x.val[curidx];
690  const SVector& A0 = A [curidx];
691  int nonzero_idx = 0;
692 
693  num = A0.size();
694  if ( isZero(x0, epsilon) || num == 0 )
695  {
696  // A[0] == 0 or x[0] == 0 => this := zero vector
697  clear();
698  }
699  else
700  {
701  for ( register int j = 0; j < num; ++j )
702  {
703  const SVector::Element& elt = A0.element(j);
704  const Real product = x0 * elt.val;
705 
706  idx[ nonzero_idx ] = elt.idx;
707  val[ elt.idx ] = product; // store the value in any case
708  if ( product != 0 ) // not 'isNotZero(product, epsilon)'
709  ++nonzero_idx; // count only non-zero values
710  }
711  }
712 
713  // Compute the other x[i] * A[i] and add them to the existing vector.
714  for ( register int i = 1; i < x.size(); ++i )
715  {
716  curidx = x.idx[i];
717  const Real xi = x.val[curidx];
718  const SVector& Ai = A [curidx];
719 
720  // If A[i] == 0 or x[i] == 0, do nothing.
721  if ( isNotZero(xi, epsilon) || Ai.size() == 0 )
722  {
723  // Compute x[i] * A[i] and add it to the existing vector.
724  for ( register int j = 0; j < Ai.size(); ++j )
725  {
726  const SVector::Element& elt = Ai.element(j);
727  idx[ nonzero_idx ] = elt.idx;
728  Real oldval = val[elt.idx];
729 
730  // An old value of exactly 0 means the position is still unused.
731  // It will be used now (either by a new nonzero or by a MARKER),
732  // so increase the counter. If oldval != 0, we just
733  // change an existing NZ-element, so don't increase the counter.
734  if ( oldval == 0 )
735  ++nonzero_idx;
736 
737  // Add the current product x[i] * A[i][j]; if oldval was
738  // MARKER before, it does not hurt because MARKER is really small.
739  oldval += xi * elt.val;
740 
741  // If the new value is exactly 0, mark the index as used
742  // by setting a value which is nearly 0; otherwise, store
743  // the value. Values below epsilon will be removed later.
744  if ( oldval == 0 )
745  val[ elt.idx ] = MARKER;
746  else
747  val[ elt.idx ] = oldval;
748  }
749  }
750  }
751 
752  // Clean up by shifting all nonzeros (w.r.t. epsilon) to the front of idx,
753  // zeroing all values which are nearly 0, and setting #num# appropriately.
754  int nz_counter = 0;
755  for ( register int i = 0; i < nonzero_idx; ++i )
756  {
757  curidx = idx[i];
758  if ( isZero( val[curidx], epsilon ) )
759  val[curidx] = 0;
760  else
761  {
762  idx[nz_counter] = curidx;
763  ++nz_counter;
764  }
765  num = nz_counter;
766  }
767 
768  assert(isConsistent());
769  return *this;
770 }
771 
773 {
774  assert(x.isSetup());
775 
776  if (x.size() == 0) // x can be setup but have size 0 => this := zero vector
777  {
778  clear();
779  return *this;
780  }
781 
782  bool A_is_zero = true;
783  for ( int i = 0; i < x.size(); ++i )
784  {
785  const int curidx = x.idx[i];
786  const Real xi = x.val[curidx];
787  const SVector& Ai = A [curidx];
788 
789  if ( A_is_zero && Ai.size() > 0 )
790  A_is_zero = false;
791 
792  for ( register int j = 0; j < Ai.size(); ++j )
793  {
794  const SVector::Element& elt = Ai.element(j);
795  val[ elt.idx ] += xi * elt.val;
796  }
797  }
798 
799  if ( A_is_zero )
800  clear(); // case x != 0 but A == 0
801 
802  return *this;
803 }
804 
806 {
807  assert(A.num() == x.dim());
808  assert(x.isSetup());
809  clear();
810 
811  if (x.size() == 1)
812  {
813  assign2product1(A, x);
814  setupStatus = true;
815  }
816 
817  else if (Real(x.size()) * A.memSize() <= shortProductFactor * dim() * A.num()
818  && isSetup())
819  {
820  assign2productShort(A, x);
821  setupStatus = true;
822  }
823 
824  else
825  {
826  assign2productFull(A, x);
827  setupStatus = false;
828  }
829  assert(isConsistent());
830 
831  return *this;
832 }
833 
835 {
836  assert(A.num() == dim());
837 
838  Real y;
839 
840  clear();
841 
842  for (int i = dim(); i-- > 0;)
843  {
844  y = A[i] * x;
845 
846  if (isNotZero(y, epsilon))
847  {
848  val[i] = y;
849  IdxSet::addIdx(i);
850  }
851  }
852  assert(isConsistent());
853 
854  return *this;
855 }
856 
858 {
859  if (x.isSetup())
860  return assign2product4setup(A, x);
861 
862 #if 0 // buggy (missing test for 'svec.size() == 0' and 'x.dim() == 0' )
863  int* xi = x.idx;
864  Real* xv = x.val;
865  Real* end = xv + x.dim() - 1;
866 
867  /* setze weissen Elefanten */
868  Real lastval = *end;
869  *end = MARKER;
870 
871  for(;;)
872  {
873  while (!*xv)
874  ++xv;
875  if (isNotZero(*xv, epsilon))
876  {
877  const SVector& svec = A[ *xi++ = int(xv - x.val) ];
878  const SVector::Element* elem = &(svec.element(0));
879  const SVector::Element* last = elem + svec.size();
880  for (; elem < last; ++elem)
881  val[elem->idx] += *xv * elem->val;
882  }
883  else
884  {
885  *xv = 0;
886  if (xv == end)
887  break;
888  }
889  xv++;
890  }
891 
892  /* fange weissen Elefanten wieder ein */
893  if (isNotZero(lastval, epsilon))
894  {
895  *xv = lastval;
896  const SVector& svec = A[ *xi++ = int(xv - x.val) ];
897  const SVector::Element* elem = &(svec.element(0));
898  const SVector::Element* last = elem + svec.size();
899  for (; elem < last; ++elem)
900  val[elem->idx] += *xv * elem->val;
901  }
902  else
903  *xv = 0;
904 
905  x.num = int(xi - x.idx);
906 
907 #else
908 
909  if (x.dim() == 0) // x == 0 => this := zero vector
910  clear();
911 
912  // x is not setup, so walk through its value vector
913  int nzcount = 0;
914  for ( register int i = 0; i < x.dim(); ++i )
915  {
916  // advance to the next element != 0
917  Real& xi = x.val[i];
918  if (xi == 0)
919  continue;
920 
921  // If x[i] is really nonzero, compute A[i] * x[i] and adapt x.idx,
922  // otherwise set x[i] to 0.
923  if (isNotZero(xi, epsilon))
924  {
925  x.idx[ nzcount++ ] = i;
926  const SVector& Ai = A[i];
927  const int Aisize = Ai.size();
928  if ( Aisize > 0 ) // otherwise: Ai == 0 => do nothing
929  {
930  for ( register int j = 0; j < Aisize; ++j )
931  {
932  const SVector::Element& elt = Ai.element(j);
933  val[elt.idx] += xi * elt.val;
934  }
935  }
936  }
937  else
938  {
939  xi = 0;
940  }
941  }
942  x.num = nzcount;
943 
944 #endif
945 
946  x.setupStatus = true;
947  setupStatus = false;
948  assert(isConsistent());
949 
950  return *this;
951 }
952 
953 
955 {
956 #ifdef ENABLE_CONSISTENCY_CHECKS
957  if (Vector::dim() > IdxSet::max())
958  return MSGinconsistent("SSVector");
959 
960  if (Vector::dim() < IdxSet::dim())
961  return MSGinconsistent("SSVector");
962 
963  if (isSetup())
964  {
965  for (int i = 0; i < Vector::dim(); ++i)
966  {
967  int j = number(i);
968 
969  if (j < 0 && fabs(val[i]) > 0.0)
970  {
971  MSG_ERROR( spxout << "ESSVEC01 i = " << i
972  << "\tidx = " << j
973  << "\tval = " << std::setprecision(16) << val[i]
974  << std::endl; )
975  return MSGinconsistent("SSVector");
976  }
977  }
978  }
980 #else
981  return true;
982 #endif
983 }
984 } // namespace soplex
985 
986 //-----------------------------------------------------------------------------
987 //Emacs Local Variables:
988 //Emacs mode:c++
989 //Emacs c-basic-offset:3
990 //Emacs tab-width:8
991 //Emacs indent-tabs-mode:nil
992 //Emacs End:
993 //-----------------------------------------------------------------------------
994