SoPlex Doxygen Documentation
dataarray.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 /**@file dataarray.h
17  * @brief Save arrays of data objects.
18  */
19 #ifndef _DATAARRAY_H_
20 #define _DATAARRAY_H_
21 
22 #include <assert.h>
23 #include <stddef.h>
24 #include <string.h>
25 #include <iostream>
26 
27 #include "spxdefines.h"
28 #include "spxalloc.h"
29 
30 namespace soplex
31 {
32 /**@brief Safe arrays of data objects.
33  @ingroup Elementary
34 
35  Class DataArray provides safe arrays of \ref DataObjects. For general
36  C++ objects (in contrast to data objects) class Array is provided which
37  manages memory in a C++ compliant way.
38 
39  The elements of an instance of DataArray can be accessed just like
40  ordinary C++ array elements by means of the index operator[](). Safety is
41  provided by
42 
43  - automatic memory management in constructor and destructor
44  preventing memory leaks
45  - checking of array bounds when accessing elements with the
46  indexing operator[]() (only when compiled without \c -DNDEBUG).
47 
48  Moreover, #DataArray%s may easily be extended by #insert%ing or #append%ing
49  elements to the DataArray or shrunken by \ref remove() "removing" elements.
50  Method reSize(int n) resets the DataArray%s length to \p n thereby possibly
51  appending elements or truncating the DataArray to the required size.
52 
53  A DataArray may be used as arguments for standard C functions requiring
54  pointers through the use of get_ptr() and get_const_ptr().
55 
56  Internally, a DataArray object allocates a block of memory that fits up
57  to max() elements, only size() of them are used. This makes extension
58  and shrinking methods perform better.
59 
60  @see Array, \ref DataObjects "Data Objects"
61 */
62 template < class T >
63 class DataArray
64 {
65 private:
66  int thesize; ///< number of used elements in array data
67  int themax; ///< the length of array data and
68  T* data; ///< the array of elements
69 
70 protected:
71  /** When a DataArray is reSize()%d to more than max() elements, the
72  new value for max() is not just set to the new size but rather to
73  \p memFactor * \p size. This makes #reSize%ing perform better in codes
74  where a DataArray is extended often by a small number of elements
75  only.
76  */
77  Real memFactor; ///< memory extension factor.
78 
79 public:
80 
81  /// reference \p n 'th element.
82  T& operator[](int n)
83  {
84  assert(n >= 0);
85  assert(n < thesize);
86  return data[n];
87  }
88  /// reference \p n 'th const element.
89  const T& operator[](int n) const
90  {
91  assert(n >= 0);
92  assert(n < thesize);
93  return data[n];
94  }
95 
96  /// reference last element.
97  T& last()
98  {
99  assert(thesize > 0);
100  return data[thesize -1];
101  }
102  /// reference last const element.
103  const T& last() const
104  {
105  assert(thesize > 0);
106  return data[thesize -1];
107  }
108 
109  /// get a C pointer to the data.
110  T* get_ptr()
111  {
112  return data;
113  }
114  /// get a const C pointer to the data.
115  const T* get_const_ptr() const
116  {
117  return data;
118  }
119 
120  /// append element \p t.
121  void append(const T& t)
122  {
123  insert(thesize, 1, &t);
124  }
125  /// append \p n elements from \p t.
126  void append(int n, const T t[])
127  {
128  insert(thesize, n, t);
129  }
130  /// append all elements from \p t.
131  void append(const DataArray<T>& t)
132  {
133  insert(thesize, t);
134  }
135 
136  /// insert \p n uninitialized elements before \p i 'th element.
137  void insert(int i, int n)
138  {
139  int j = thesize;
140 
141  assert(i >= 0);
142  assert(n >= 0);
143 
144  reSize(thesize + n);
145 
146  /// move \p n elements in memory from insert position \p i to the back
147  if( j > i )
148  memmove(&(data[i+n]), &(data[i]), j - i);
149  }
150 
151  /// insert \p n elements from \p t before \p i 'the element.
152  void insert(int i, int n, const T t[])
153  {
154  if (n > 0)
155  {
156  insert(i, n);
157  memcpy(&(data[i]), t, n*sizeof(T));
158  }
159  }
160 
161  /// insert all elements from \p t before \p i 'th element.
162  void insert(int i, const DataArray<T>& t)
163  {
164  if (t.size())
165  {
166  insert(i, t.size());
167  memcpy(&(data[i]), t.data, t.size()*sizeof(T));
168  }
169  }
170 
171  /// remove \p m elements starting at \p n.
172  void remove(int n = 0, int m = 1)
173  {
174  assert(n < size() && n >= 0);
175  /* use memmove instead of memcopy because the destination and the source might overlap */
176  if (n + m < size())
177  memmove(&(data[n]), &(data[n + m]), (size() - (n + m)) * sizeof(T));
178  else
179  m = size() - n;
180  thesize -= m;
181  }
182  /// remove \p m last elements.
183  void removeLast(int m = 1)
184  {
185  assert(m <= size() && m >= 0);
186  thesize -= m;
187  }
188  /// remove all elements.
189  void clear()
190  {
191  thesize = 0;
192  }
193 
194  /// return nr. of elements.
195  int size() const
196  {
197  return thesize;
198  }
199 
200  /// reset size to \p newsize.
201  /** Resizing a DataArray to less than the previous size, involves
202  discarding its last elements. Resizing to a larger value involves
203  adding uninitialized elements (similar to append()). If neccessary,
204  also memory will be reallocated.
205  @param newsize the new number of elements the array can hold.
206  */
207  void reSize(int newsize)
208  {
209  assert(memFactor >= 1);
210  if (newsize > themax)
211  reMax(int(memFactor * newsize), newsize);
212  else if (newsize < 0)
213  thesize = 0;
214  else
215  thesize = newsize;
216  }
217 
218  /// return maximum number of elements.
219  /** Even though the DataArray currently holds no more than size()
220  elements, up to max() elements could be added without need to
221  reallocated free store.
222  */
223  int max() const
224  {
225  return themax;
226  }
227 
228  /// reset maximum number of elements.
229  /** The value of max() is reset to \p newMax thereby setting size()
230  to \p newSize. However, if \p newSize has a value \c < \c 0 (as the
231  default argument does) size() remains unchanged and max() is set
232  to MIN(size(), newMax). Hence, calling reMax() without the
233  default arguments, will reduce the memory consumption to a minimum.
234  In no instance max() will be set to a value less than 1 (even if
235  specified).
236 
237  @return reMax returns the difference in bytes of the new and the old
238  memory block, which can be used to update pointers pointing
239  to elements of the memory block.
240  */
241  ptrdiff_t reMax(int newMax = 1, int newSize = -1)
242  {
243  if (newSize >= 0)
244  thesize = newSize;
245  if (newMax < newSize)
246  newMax = newSize;
247  if (newMax < 1)
248  newMax = 1;
249  if (newMax == themax)
250  return 0;
251  themax = newMax;
252  T* olddata = data;
253  if (thesize <= 0)
254  {
255  spx_free(data);
257  }
258  else
260 
261  return reinterpret_cast<char*>(data) - reinterpret_cast<char*>(olddata);
262  }
263  /// assignment operator
265  {
266  if (this != &rhs)
267  {
268  reSize(rhs.size());
269  memcpy(data, rhs.data, size() * sizeof(T));
270 
271  assert(isConsistent());
272  }
273  return *this;
274  }
275 
276  /// consistency check
277  bool isConsistent() const
278  {
279 #ifdef ENABLE_CONSISTENCY_CHECKS
280  if ( (data == 0)
281  || (themax < 1)
282  || (themax < thesize)
283  || (thesize < 0)
284  || (memFactor < 1.0))
285  return MSGinconsistent("DataArray");
286 #endif
287 
288  return true;
289  }
290 
291  /// copy constructor
292  DataArray(const DataArray& old)
293  : thesize(old.thesize)
294  , themax (old.themax)
295  , data (0)
296  , memFactor (old.memFactor)
297  {
298  spx_alloc(data, max());
299 
300  if (thesize)
301  memcpy(data, old.data, thesize * sizeof(T));
302 
303  assert(isConsistent());
304  }
305 
306  /// default constructor.
307  /** The constructor allocates an Array containing \p size uninitialized
308  elements. The internal array is allocated to have \p max nonzeros,
309  and the memory extension factor is set to \p fac.
310 
311  @param p_size number of unitialised elements.
312  @param p_max maximum number of elements the array can hold.
313  @param p_fac value for memFactor.
314  */
315  explicit DataArray(int p_size = 0, int p_max = 0, Real p_fac = 1.2)
316  : data (0)
317  , memFactor(p_fac)
318  {
319  thesize = (p_size < 0) ? 0 : p_size;
320  if (p_max > thesize)
321  themax = p_max;
322  else
323  themax = (thesize == 0) ? 1 : thesize;
324 
326 
327  assert(isConsistent());
328  }
329 
330  /// destructor
332  {
333  if(data)
334  spx_free(data);
335  }
336 };
337 
338 } // namespace soplex
339 #endif // _DATAARRAY_H_
340 
341 //-----------------------------------------------------------------------------
342 //Emacs Local Variables:
343 //Emacs mode:c++
344 //Emacs c-basic-offset:3
345 //Emacs tab-width:8
346 //Emacs indent-tabs-mode:nil
347 //Emacs End:
348 //-----------------------------------------------------------------------------