Simpatico  v1.10
GPArray.h
1 #ifndef UTIL_G_P_ARRAY_H
2 #define UTIL_G_P_ARRAY_H
3 
4 /*
5 * Util Package - C++ Utilities for Scientific Computation
6 *
7 * Copyright 2010 - 2017, The Regents of the University of Minnesota
8 * Distributed under the terms of the GNU General Public License.
9 */
10 
11 #include <util/containers/PArray.h>
12 #include <util/global.h>
13 
14 namespace Util
15 {
16 
27  template <typename Data>
28  class GPArray : public PArray<Data>
29  {
30 
31  public:
32 
36  GPArray();
37 
45  GPArray(const GPArray<Data>& other);
46 
56  GPArray<Data>& operator=(const GPArray<Data>& other);
57 
64  virtual ~GPArray();
65 
73  void append(Data& data);
74 
84  void reserve(int capacity);
85 
89  void deallocate();
90 
94  void clear();
95 
99  bool isAllocated() const;
100 
101  protected:
102 
103  using PArray<Data>::ptrs_;
105  using PArray<Data>::size_;
106 
107  };
108 
109  /*
110  * Default constructor.
111  */
112  template <typename Data>
114  : PArray<Data>()
115  {}
116 
124  template <typename Data>
126  : PArray<Data>()
127  {
128  assert(other.capacity_ >= other.size_);
129  if (other.ptrs_ == 0) {
130  assert(other.capacity_ == 0);
131  assert(other.size_ == 0);
132  ptrs_ = 0;
133  capacity_ = 0;
134  size_ = 0;
135  } else {
136  assert(other.capacity_ > 0);
137  // Allocate array of Data* pointers
138  Memory::allocate<Data*>(ptrs_, other.capacity_);
139  capacity_ = other.capacity_;
140  size_ = other.size_;
141  // Copy pointers
142  if (size_ > 0) {
143  for (int i = 0; i < size_; ++i) {
144  ptrs_[i] = other.ptrs_[i];
145  }
146  }
147  // Nullify unused elements of ptrs_ array
148  if (capacity_ > size_) {
149  for (int i = size_; i < capacity_; ++i) {
150  ptrs_[i] = 0;
151  }
152  }
153  }
154  }
155 
156  /*
157  * Assignment, element by element.
158  */
159  template <typename Data>
161  {
162  // Check for self assignment
163  if (this == &other) return *this;
164 
165  clear();
166  for (int i = 0; i < other.size_; ++i) {
167  append(other[i]);
168  }
169  return *this;
170  }
171 
172  /*
173  * Destructor.
174  */
175  template <typename Data>
177  {
178  size_ = 0;
179  if (isAllocated()) {
181  capacity_ = 0;
182  }
183  }
184 
185  /*
186  * Reserve space for the underlying array of Data* pointers.
187  */
188  template <typename Data>
190  {
191  if (capacity <= 0) {
192  UTIL_THROW("Cannot reserve with capacity <=0");
193  }
194  if (ptrs_ == 0) {
195  assert(capacity_ == 0);
196  assert(size_ == 0);
197  Memory::allocate<Data*>(ptrs_, capacity);
199  size_ = 0;
200  } else if (capacity > capacity_) {
201  assert(capacity_ > 0);
202  assert(capacity_ >= size_);
203  assert(size_ >= 0);
204  Data** newPtr = 0;
205  Memory::allocate<Data*>(newPtr, capacity);
206  if (size_ > 0) {
207  for (int i = 0; i < size_; ++i) {
208  newPtr[i] = ptrs_[i];
209  }
210  }
211  Memory::deallocate<Data*>(ptrs_, capacity_);
212  ptrs_ = newPtr;
214  }
215  }
216 
217  /*
218  * Deallocate associated memory.
219  */
220  template <typename Data>
222  {
223  size_ = 0;
224  if (isAllocated()) {
225  assert(capacity_ > 0);
226  Memory::deallocate<Data*>(ptrs_, capacity_);
227  capacity_ = 0;
228  }
229  }
230 
231  /*
232  * Append an element to the end of the PArray.
233  */
234  template <typename Data>
235  void GPArray<Data>::append(Data& data)
236  {
237  if (!isAllocated()) {
238  assert(capacity_ == 0);
239  Memory::allocate<Data*>(ptrs_, 64);
240  capacity_ = 64;
241  size_ = 0;
242  } else if (size_ == capacity_) {
243  assert(capacity_ > 0);
244  assert(capacity_ >= size_);
245  assert(size_ >= 0);
246  Data** newPtr = 0;
247  Memory::allocate<Data*>(newPtr, 2*capacity_);
248  if (size_ > 0) {
249  for (int i = 0; i < size_; ++i) {
250  newPtr[i] = ptrs_[i];
251  }
252  }
253  Memory::deallocate<Data*>(ptrs_, capacity_);
254  ptrs_ = newPtr;
255  capacity_ = 2*capacity_;
256  // size_ is unchanged
257  }
258  // Append new element
259  ptrs_[size_] = &data;
260  ++size_;
261  assert(size_ <= capacity_);
262  }
263 
264  /*
265  * Reset to empty state, without deallocating.
266  */
267  template <typename Data>
268  inline void GPArray<Data>::clear()
269  { size_ = 0; }
270 
271  /*
272  * Is this GPArray allocated?
273  */
274  template <class Data>
275  inline bool GPArray<Data>::isAllocated() const
276  { return (bool)ptrs_; }
277 
278 }
279 #endif
void reserve(int capacity)
Reserve memory for specified number of elements.
Definition: GPArray.h:189
int capacity_
Allocated size of ptrs_ array.
Definition: PArray.h:90
int capacity() const
Return allocated size.
Definition: PArray.h:130
File containing preprocessor macros for error handling.
#define UTIL_THROW(msg)
Macro for throwing an Exception, reporting function, file and line number.
Definition: global.h:51
Utility classes for scientific computation.
Definition: accumulators.mod:1
int size_
Logical size (number of elements with initialized data).
Definition: PArray.h:93
GPArray< Data > & operator=(const GPArray< Data > &other)
Assignment, element by element.
Definition: GPArray.h:160
void clear()
Reset to empty state.
Definition: GPArray.h:268
GPArray()
Constructor.
Definition: GPArray.h:113
Data ** ptrs_
PArray of of pointers to Data objects.
Definition: PArray.h:87
void deallocate()
Deallocate (delete) underlying array of pointers.
Definition: GPArray.h:221
bool isAllocated() const
Is this GPArray allocated?
Definition: GPArray.h:275
An automatically growable PArray.
Definition: GPArray.h:28
virtual ~GPArray()
Destructor.
Definition: GPArray.h:176
void append(Data &data)
Append an element to the end of the sequence.
Definition: GPArray.h:235
static void deallocate(Data *&ptr, size_t size)
Allocate a C array.
Definition: Memory.h:124
An array that only holds pointers to its elements.
Definition: PArray.h:33