SoPlex Doxygen Documentation
ssvector.h
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 
17 /**@file ssvector.h
18  * @brief Semi sparse vector.
19  */
20 #ifndef _SSVECTOR_H_
21 #define _SSVECTOR_H_
22 
23 #include <assert.h>
24 
25 #include "spxdefines.h"
26 #include "dvector.h"
27 #include "svector.h"
28 #include "didxset.h"
29 #include "spxalloc.h"
30 
31 namespace soplex
32 {
33 class SVSet;
34 
35 /**@brief Semi sparse vector.
36  @ingroup Algebra
37 
38  This class implements Semi Sparse Vectors. Such are
39  #DVector%s where the indices of its nonzero elements can be stored in an
40  extra IdxSet. Only elements with absolute value > #epsilon are considered
41  to be nonzero.
42  Since really storing the nonzeros is not always convenient,
43  an SSVector provides two different stati: setup and not setup.
44  An SSVector being setup means that the nonzero indices are available,
45  otherwise an SSVector is just an ordinary Vector with an empty IdxSet.
46  Note that due to arithmetic operation, zeros can slip in, i.e., it is only
47  guaranteed that at least every non-zero is in the IdxSet.
48 */
49 class SSVector : protected DVector, protected IdxSet
50 {
51 private:
52 
53  friend class DVector;
54  friend class Vector;
55  friend class DSVector;
56 
57  //--------------------------------------------
58  /**@name Data */
59  //@{
60  /// Is the SSVector set up?
62  /// Allocates enough space to accommodate \p newmax values.
63  void setMax(int newmax);
64  /// A value x with |x| < epsilon is considered zero.
66  //@}
67 
68 public:
69 
70  //--------------------------------------------
71  /**@name Status of an SSVector
72  An SSVector can be set up or not. In case it is set up, its IdxSet
73  correctly contains all indices of nonzero elements of the SSVector.
74  Otherwise, it does not contain any useful data. Whether or not an
75  SSVector is setup can be determined with the method
76  \ref soplex::SSVector::isSetup() "isSetup()".
77 
78  There are three methods for directly affecting the setup status of an
79  SSVector:
80  - unSetup(): This method sets the status to ``not setup''.
81  - setup(): This method initializes the IdxSet to the
82  SSVector's nonzero indices and sets the status
83  to ``setup''.
84  - forceSetup(): This method sets the status to ``setup'' without
85  verifying that the IdxSet correctly contains
86  all nonzero indices. It may be used when the
87  nonzero indices have been computed externally.
88  */
89  //@{
90  /// only used in slufactor.cpp
92  {
93  return DVector::get_ptr();
94  }
95  /// returns the non-zero epsilon used.
96  Real getEpsilon() const
97  {
98  return epsilon;
99  }
100  /// sets the non-zero epsilon.
101  /** This invalidates the setup.
102  */
103  void setEpsilon(Real eps)
104  {
105  if( eps != epsilon )
106  {
107  epsilon = eps;
108  setupStatus = false;
109  }
110  }
111  /// returns setup status.
112  bool isSetup() const
113  {
114  return setupStatus;
115  }
116 
117  /// makes SSVector not setup.
118  void unSetup()
119  {
120  setupStatus = false;
121  }
122 
123  /// initializes nonzero indices
124  /** Initializes nonzero indices for all elements with absolute values
125  greater than #epsilon and sets all other elements to 0.
126  */
127  void setup();
128 
129  /// forces setup status.
130  void forceSetup()
131  {
132  setupStatus = true;
133  }
134  //@}
135 
136 
137  //--------------------------------------------
138  /**@name Methods for setup SSVectors */
139  //@{
140  /// returns index of the \p n 'th nonzero element.
141  int index(int n) const
142  {
143  assert(isSetup());
144  return IdxSet::index(n);
145  }
146 
147  /// returns value of the \p n 'th nonzero element.
148  Real value(int n) const
149  {
150  assert(isSetup());
151  assert(n >= 0 && n < size());
152  return val[idx[n]];
153  }
154 
155  /// returns the position number of index \p i, or -1 if \p i doesn't exist.
156  int number(int i) const
157  {
158  assert(isSetup());
159  return IdxSet::number(i);
160  }
161 
162  /// returns the number of nonzeros.
163  int size() const
164  {
165  assert(isSetup());
166  return IdxSet::size();
167  }
168 
169  /// adds nonzero (\p i, \p x) to SSVector.
170  /** No nonzero with index \p i must exist in the SSVector.
171  */
172  void add(int i, Real x)
173  {
174  assert(val[i] == 0);
175  assert(number(i) < 0);
176  addIdx(i);
177  val[i] = x;
178  }
179 
180  /// sets \p i 'th element to \p x.
181  void setValue(int i, Real x);
182 
183  /// clears element \p i.
184  void clearIdx(int i)
185  {
186  if (isSetup())
187  {
188  int n = number(i);
189  if (n >= 0)
190  remove(n);
191  }
192  val[i] = 0;
193 
194  assert(isConsistent());
195  }
196 
197  /// sets \p n 'th nonzero element to 0 (index \p n must exist!).
198  void clearNum(int n)
199  {
200  assert(isSetup());
201  assert(index(n) >= 0);
202  val[index(n)] = 0;
203  remove(n);
204 
205  assert(isConsistent());
206  }
207  //@}
208 
209 
210  //--------------------------------------------
211  /**@name Methods independent of the Status */
212  //@{
213  /// returns \p i 'th value.
214  Real operator[](int i) const
215  {
216  return val[i];
217  }
218 
219  /// returns array indices.
220  const int* indexMem() const
221  {
222  return idx;
223  }
224 
225  /// returns array values.
226  const Real* values() const
227  {
228  return val;
229  }
230 
231  /// returns indices.
232  const IdxSet& indices() const
233  {
234  return *this;
235  }
236 
237  /// returns array indices.
238  int* altIndexMem()
239  {
240  unSetup();
241  return idx;
242  }
243 
244  /// returns array values.
246  {
247  unSetup();
248  return val;
249  }
250 
251  /// returns indices.
253  {
254  unSetup();
255  return *this;
256  }
257  //@}
258 
259 
260  //------------------------------------
261  /**@name Mathematical operations */
262  //@{
263  ///
264  SSVector& operator+=(const Vector& vec);
265  ///
266  SSVector& operator+=(const SVector& vec);
267  /// vector summation.
268  SSVector& operator+=(const SSVector& vec);
269 
270  ///
271  SSVector& operator-=(const Vector& vec);
272  ///
273  SSVector& operator-=(const SVector& vec);
274  /// vector subtraction.
275  SSVector& operator-=(const SSVector& vec);
276 
277  /// vector scaling.
279 
280  ///
281  //SSVector& multAdd(Real x, const SSVector& vec);
282  ///
283  SSVector& multAdd(Real x, const SVector& vec);
284  /// adds scaled vector (+= \p x * \p vec).
285  SSVector& multAdd(Real x, const Vector& vec);
286  /// assigns SSVector to \f$x^T \cdot A\f$.
287  SSVector& assign2product(const SSVector& x, const SVSet& A);
288  /// assigns SSVector to \f$A \cdot x\f$ for a setup \p x.
289  SSVector& assign2product4setup(const SVSet& A, const SSVector& x);
290 
291 public:
292 
293  /// assigns SSVector to \f$A \cdot x\f$ thereby setting up \p x.
295 
296  /// returns infinity norm of a Vector.
297  Real maxAbs() const;
298  /// returns euclidian norm of a Vector.
299  Real length() const;
300  /// returns squared norm of a Vector.
301  Real length2() const;
302  //@}
303 
304 
305  //------------------------------------
306  /**@name Miscellaneous */
307  //@{
308  /// returns dimension of Vector.
309  int dim() const
310  {
311  return dimen;
312  }
313 
314  /// resets dimension to \p newdim.
315  void reDim (int newdim);
316 
317  /// sets number of nonzeros (thereby unSetup SSVector).
318  void setSize(int n)
319  {
320  assert(n >= 0);
321  assert(n <= IdxSet::max());
322  unSetup();
323  num = n;
324  }
325 
326  /// resets memory consumption to \p newsize.
327  void reMem(int newsize);
328 
329  /// clears vector.
330  void clear ();
331 
332  /// consistency check.
333  bool isConsistent() const;
334  //@}
335 
336 
337  //------------------------------------
338  /**@name Constructors / Destructors */
339  //@{
340  /// constructor.
341  explicit SSVector(int p_dim, Real p_eps = Param::epsilon())
342  : DVector (p_dim)
343  , IdxSet ()
344  , setupStatus(true)
345  , epsilon (p_eps)
346  {
347  len = (p_dim < 1) ? 1 : p_dim;
348  spx_alloc(idx, len);
349 
350  Vector::clear();
351 
352  assert(isConsistent());
353  }
354 
355  /// copy constructor.
356  SSVector(const SSVector& vec)
357  : DVector (vec)
358  , IdxSet ()
359  , setupStatus(vec.setupStatus)
360  , epsilon (vec.epsilon)
361  {
362  len = (vec.dim() < 1) ? 1 : vec.dim();
363  spx_alloc(idx, len);
364  IdxSet::operator= ( vec );
365 
366  assert(isConsistent());
367  }
368 
369  /// constructs nonsetup copy of \p vec.
370  explicit SSVector(const Vector& vec, Real eps = Param::epsilon())
371  : DVector (vec)
372  , IdxSet ()
373  , setupStatus(false)
374  , epsilon (eps)
375  {
376  len = (vec.dim() < 1) ? 1 : vec.dim();
377  spx_alloc(idx, len);
378  /// TODO: @todo Is there an IdxSet::operator=( vec ) missing here?
379  assert(isConsistent());
380  }
381 
382  /// sets up \p rhs vector, and assigns it.
383  void setup_and_assign(SSVector& rhs);
384 
385  /// assign only the elements of \p rhs.
386  SSVector& assign(const SVector& rhs);
387  /// assignment operator
388  SSVector& operator=(const SSVector& rhs);
389  /// assignment operator
390  SSVector& operator=(const SVector& rhs);
391  /// assignment operator
392  SSVector& operator=(const Vector& rhs)
393  {
394  unSetup();
395  Vector::operator=(rhs);
396 
397  assert(isConsistent());
398  return *this;
399  }
400  /// destructor
402  {
403  if ( idx )
404  spx_free(idx);
405  }
406  //@}
407 
408 private:
409 
410  //----------------------------
411  /**@name Private helpers */
412  //@{
413  ///
414  SSVector& assign2product1(const SVSet& A, const SSVector& x);
415  ///
416  SSVector& assign2productShort(const SVSet& A, const SSVector& x);
417  ///
418  SSVector& assign2productFull(const SVSet& A, const SSVector& x);
419  //@}
420 };
421 
422 
423 // ----------------------------------------------------------------------------
424 // Vector operators involving SSVectors
425 // ----------------------------------------------------------------------------
426 
427 inline Vector& Vector::multAdd(Real x, const SSVector& svec)
428 {
429  assert(svec.dim() <= dim());
430 
431  if (svec.isSetup())
432  {
433  const int* idx = svec.indexMem();
434 
435  for(int i = 0; i < svec.size(); i++)
436  val[idx[i]] += x * svec[idx[i]];
437  }
438  else
439  {
440  assert(svec.dim() == dim());
441 
442  for(int i = 0; i < dim(); i++)
443  val[i] += x * svec.val[i];
444  }
445  //multAdd(x, static_cast<const Vector&>(svec));
446 
447  return *this;
448 }
449 
450 inline Vector& Vector::assign(const SSVector& svec)
451 {
452  assert(svec.dim() <= dim());
453 
454  if (svec.isSetup())
455  {
456  const int* idx = svec.indexMem();
457 
458  for(int i = svec.size(); i > 0; i--)
459  {
460  val[*idx] = svec.val[*idx];
461  idx++;
462  }
463  }
464  else
465  operator= (static_cast<const Vector&>(svec));
466 
467  return *this;
468 }
469 
470 inline Vector& Vector::operator=(const SSVector& vec)
471 {
472  if (vec.isSetup())
473  {
474  clear ();
475  assign(vec);
476  }
477  else
478  operator= (static_cast<const Vector&>(vec));
479 
480  return *this;
481 }
482 
483 inline Real Vector::operator*(const SSVector& v) const
484 {
485  assert(dim() == v.dim());
486 
487  if (v.isSetup())
488  {
489  const int* idx = v.indexMem();
490  Real x = 0;
491 
492  for(int i = v.size(); i > 0; i--)
493  {
494  x += val[*idx] * v.val[*idx];
495  idx++;
496  }
497  return x;
498  }
499  else
500  return operator*(static_cast<const Vector&>(v));
501 }
502 } // namespace soplex
503 #endif // _SSVECTOR_H_
504 
505 //-----------------------------------------------------------------------------
506 //Emacs Local Variables:
507 //Emacs mode:c++
508 //Emacs c-basic-offset:3
509 //Emacs tab-width:8
510 //Emacs indent-tabs-mode:nil
511 //Emacs End:
512 //-----------------------------------------------------------------------------