SoPlex Doxygen Documentation
array.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 array.h
17  * @brief Save arrays of arbitrary types.
18  */
19 #ifndef _ARRAY_H_
20 #define _ARRAY_H_
21 
22 #include <assert.h>
23 
24 
25 namespace soplex
26 {
27 /**@brief Safe arrays of arbitrary types.
28  @ingroup Elementary
29 
30  Class Array provides safe arrays of arbitrary type. Array elements are
31  accessed just like ordinary C++ array elements by means of the index
32  operator[](). Safety is provided by
33 
34  - automatic memory management in constructor and destructure
35  preventing memory leaks
36  - checking of array bound when accessing elements with the
37  indexing operator[]() (only when compiled without \c -DNDEBUG).
38 
39  Moreover, #Array%s may easily be extended by #insert%ing or
40  #append%ing elements to the Array or shrunken by
41  \ref remove() "removing"
42  elements. Method reSize(int n) resets the Array's length to \p n,
43  thereby appending elements or truncating the Array to the
44  required size.
45 
46  An Array is implemented in a C++-compliant way with respect to
47  how memory is managed: Only operators new and delete are
48  used for allocating memory. This involves some overhead for all
49  methods effecting the length of an Array, i.e., all methods
50  insert(), append(), remove() and reSize(). This involves
51  allocating a new C++ array of the new size and copying all
52  elements with the template parameters operator=().
53 
54  For this reason, it is not convenient to use class Array if its elements
55  are \ref DataObjects "Data Objects". In this case use class DataArray
56  instead.
57 
58  @see DataArray, \ref DataObjects "Data Objects"
59 */
60 template < class T >
61 class Array
62 {
63 protected:
64 
65  //----------------------------------------
66  /**@name Data */
67  //@{
68  int num; ///< the length of array data
69  T* data; ///< the array of elements
70  //@}
71 
72 public:
73 
74  //----------------------------------------
75  /**@name Access / modification */
76  //@{
77  /// reference \p n 'th element.
78  T& operator[](int n)
79  {
80  assert(n >= 0 && n < size());
81  return data[n];
82  }
83  /// reference \p n 'th element.
84  const T& operator[](int n) const
85  {
86  assert(n >= 0 && n < size());
87  return data[n];
88  }
89 
90  /** This function serves for using a Vector in an C-style
91  * function. It returns a pointer to the first value of the array.
92  */
93  T* get_ptr()
94  {
95  return data;
96  }
97 
98  /// append \p n uninitialized elements.
99  void append(int n)
100  {
101  insert(size(), n);
102  }
103  /// append \p n elements from \p p_array.
104  void append(int n, const T* p_array)
105  {
106  insert(size(), n, p_array);
107  }
108  /// append all elements from \p p_array.
109  void append(const Array<T>& p_array)
110  {
111  insert(size(), p_array);
112  }
113 
114  /// insert \p n uninitialized elements before \p i 'th element.
115  void insert(int i, int n)
116  {
117  assert(i <= size());
118  if (n > 0)
119  {
120  int k;
121  T *olddata = data;
122  data = new T[size() + n]();
123  assert(data != 0);
124  if (size() > 0)
125  {
126  for (k = 0; k < i; ++k)
127  data[k] = olddata[k];
128  for (; k < size(); ++k)
129  data[k + n] = olddata[k];
130  delete[] olddata;
131  }
132  num += n;
133  }
134  }
135 
136  /// insert \p n elements from \p p_array before \p i 'th element.
137  void insert(int i, int n, const T* p_array)
138  {
139  insert(i, n);
140  for (n--; n >= 0; --n)
141  data[n + i] = p_array[n];
142  }
143 
144  /// insert all elements from \p p_array before \p i 'th element.
145  void insert(int i, const Array<T>& p_array)
146  {
147  int n = p_array.size();
148  insert(i, n);
149  for (n--; n >= 0; --n)
150  data[n + i] = p_array.data[n];
151  }
152 
153  /// remove \p m elements starting at \p n.
154  void remove(int n = 0, int m = 1)
155  {
156  assert(n >= 0 && m >= 0);
157  if (m > 0 && n < size())
158  {
159  T *olddata = data;
160  m -= (n + m <= size()) ? 0 : n + m - size();
161  num -= m;
162  if (num > 0)
163  {
164  int i;
165  data = new T[num]();
166  for (i = 0; i < n; ++i)
167  data[i] = olddata[i];
168  for (; i < num; ++i)
169  data[i] = olddata[i + m];
170  }
171  delete[] olddata;
172  }
173  }
174 
175  /// remove all elements.
176  void clear()
177  {
178  if (num > 0)
179  {
180  num = 0;
181  delete[] data;
182  }
183  }
184 
185  /// return the number of elements.
186  int size() const
187  {
188  return num;
189  }
190 
191  /// reset the number of elements.
192  void reSize(int newsize)
193  {
194  if (newsize < size())
195  remove(newsize, size() - newsize);
196  else if (newsize > size())
197  append(newsize - size());
198  }
199  //@}
200 
201  //----------------------------------------
202  /**@name Construction / destruction */
203  //@{
204  /// assignment operator.
205  /** Assigning an rvalue Array to an lvalue Array involves resizing
206  * the lvalue to the rvalues size() and copying all elements via
207  * the Array element's assignment operator=().
208  */
210  {
211  if (this != &rhs)
212  {
213  reSize(rhs.size());
214  for (int i = 0; i < size(); ++i)
215  data[i] = rhs.data[i];
216  assert(Array::isConsistent());
217  }
218  return *this;
219  }
220 
221  /// default constructor.
222  /** The constructor allocates an Array of \p n uninitialized elements.
223  */
224  explicit
225  Array(int n = 0)
226  : data(0)
227  {
228  assert(n >= 0);
229  num = n;
230  if (num > 0)
231  {
232  data = new T[num]();
233  assert(data != 0);
234  }
235  assert(Array::isConsistent());
236  }
237 
238  /// copy constructor
239  Array(const Array<T>& old)
240  : num(old.num)
241  {
242  if (num > 0)
243  {
244  data = new T[num]();
245  assert(data != 0);
246  *this = old;
247  }
248  else
249  data = 0;
250  assert(Array::isConsistent());
251  }
252 
253  /// destructor
255  {
256  if (num > 0)
257  delete[] data;
258  }
259 
260  /// consistency check
261  bool isConsistent() const
262  {
263 #ifdef ENABLE_CONSISTENCY_CHECKS
264  if (num < 0 || (num > 0 && data == 0))
265  return MSGinconsistent("Array");
266 #endif
267 
268  return true;
269  }
270  //@}
271 };
272 } // namespace soplex
273 #endif // _ARRAY_H_
274 
275 //-----------------------------------------------------------------------------
276 //Emacs Local Variables:
277 //Emacs mode:c++
278 //Emacs c-basic-offset:3
279 //Emacs tab-width:8
280 //Emacs indent-tabs-mode:nil
281 //Emacs End:
282 //-----------------------------------------------------------------------------