Simpatico  v1.10
ArrayStack.h
1 #ifndef UTIL_ARRAY_STACK_H
2 #define UTIL_ARRAY_STACK_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/misc/Memory.h>
12 #include <util/global.h>
13 
14 namespace Util
15 {
16 
24  template <typename Data>
25  class ArrayStack
26  {
27 
28  public:
29 
33  ArrayStack();
34 
38  virtual ~ArrayStack();
39 
45  void allocate(int capacity);
46 
48 
49 
55  void push(Data& data);
56 
62  Data& pop();
63 
65 
67 
73  int capacity() const;
74 
78  int size() const;
79 
83  Data& peek();
84 
88  const Data& peek() const;
89 
93  bool isValid() const;
94 
98  bool isAllocated() const;
99 
101 
102  private:
103 
104  // C array of pointers to elements of data_ array
105  Data **ptrs_;
106 
107  // Allocated size of ptrs_ array (maximum size of stack).
108  int capacity_;
109 
110  // Size of stack.
111  int size_;
112 
114  ArrayStack(const ArrayStack&);
115 
117  ArrayStack& operator = (const ArrayStack&);
118 
119  };
120 
121 
122  /*
123  * Default constructor.
124  */
125  template <typename Data>
127  : ptrs_(0),
128  capacity_(0),
129  size_(0)
130  {}
131 
132  /*
133  * Destructor.
134  */
135  template <typename Data>
137  {
138  if (ptrs_) {
139  Memory::deallocate<Data*>(ptrs_, capacity_);
140  }
141  }
142 
143  /*
144  * Allocate and initialize required memory.
145  */
146  template <typename Data>
148  {
149  if (capacity <=0) {
150  UTIL_THROW("Cannot allocate ArrayStack with capacity <= 0");
151  }
152  if (ptrs_) {
153  UTIL_THROW("Attempt to re-allocate an ArrayStack");
154  }
155  Memory::allocate<Data*>(ptrs_, capacity);
156  capacity_ = capacity;
157  size_ = 0;
158 
159  // Nullify all Data* pointer elements
160  for (int i = 0; i < capacity_; ++i) {
161  ptrs_[i] = 0;
162  }
163  }
164 
165  /*
166  * Return capacity of the associated Array.
167  */
168  template <typename Data>
170  { return capacity_; }
171 
172  /*
173  * Get the current size.
174  */
175  template <typename Data>
176  inline int ArrayStack<Data>::size() const
177  { return size_; }
178 
179  /*
180  * Push a Data object onto the stack.
181  */
182  template <typename Data>
183  void ArrayStack<Data>::push(Data& data)
184  {
185  if (size_ == capacity_) {
186  UTIL_THROW("Attempt to push onto full stack");
187  }
188  ptrs_[size_] = &data;
189  ++size_;
190  }
191 
192  /*
193  * Pop a Data object off the stack.
194  */
195  template <typename Data>
197  {
198  if (size_ == 0) {
199  UTIL_THROW("Attempt to pop from empty stack");
200  }
201  Data *ptr = ptrs_[size_-1];
202  ptrs_[size_-1] = 0;
203  --size_;
204  return *ptr;
205  }
206 
207  /*
208  * Return a reference to the top element, without popping.
209  */
210  template <typename Data>
211  inline Data& ArrayStack<Data>::peek()
212  { return *ptrs_[size_-1]; }
213 
214  /*
215  * Return a const reference to the top element, without popping.
216  */
217  template <typename Data>
218  inline const Data& ArrayStack<Data>::peek() const
219  { return *ptrs_[size_-1]; }
220 
221  /*
222  * Return true if valid, or throw an exception.
223  */
224  template <typename Data>
226  {
227  if (capacity_ < 0) {
228  UTIL_THROW("Negative capacity_");
229  }
230  if (size_ < 0) {
231  UTIL_THROW("Negative size_");
232  }
233  if (size_ > capacity_) {
234  UTIL_THROW("size_ > capacity_");
235  }
236 
237  if (ptrs_ != 0) {
238  int i;
239  for (i = 0; i < size_ ; ++i) {
240  if (ptrs_[i] == 0) {
241  UTIL_THROW("Null ptrs_[i] for i < size_");
242  }
243  }
244  for (i = size_; i < capacity_ ; ++i) {
245  if (ptrs_[i] != 0) {
246  UTIL_THROW("Non-null ptrs_[i] for i >= size_");
247  }
248  }
249  } else {
250  if (capacity_ != 0) {
251  UTIL_THROW("Unallocated stack but capacity != 0");
252  }
253  }
254  return true;
255  }
256 
257  /*
258  * Return true if the ArrayStack has been allocated, false otherwise.
259  */
260  template <typename Data>
261  inline bool ArrayStack<Data>::isAllocated() const
262  { return (ptrs_ != 0); }
263 
264 }
265 #endif
int capacity() const
Return capacity of the underlying array.
Definition: ArrayStack.h:169
ArrayStack()
Default constructor.
Definition: ArrayStack.h:126
virtual ~ArrayStack()
Destructor.
Definition: ArrayStack.h:136
Data & peek()
Return a reference to the top element (don&#39;t pop).
Definition: ArrayStack.h:211
bool isValid() const
Return true if the ArrayStack is valid, or throw an exception.
Definition: ArrayStack.h:225
File containing preprocessor macros for error handling.
void push(Data &data)
Push an element onto the Stack.
Definition: ArrayStack.h:183
int size() const
Get the number of elements in the stack.
Definition: ArrayStack.h:176
Data & pop()
Pop an element off the stack.
Definition: ArrayStack.h:196
#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
A stack of fixed capacity.
Definition: ArrayStack.h:25
void allocate(int capacity)
Initialize and allocate required memory.
Definition: ArrayStack.h:147
bool isAllocated() const
Return true only if the ArrayStack has been allocated.
Definition: ArrayStack.h:261