-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsmarter_stack_lib.cc
198 lines (175 loc) · 4.8 KB
/
smarter_stack_lib.cc
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
#include "smarter_stack.hh"
#include <cerrno>
#include <limits>
using namespace std;
namespace smarter_stack {
namespace {
// One-element stack is defined as increasing.
bool is_increasing(const SmarterStack &stack) {
// depth() returns number of elements in array indexed from 0 to top_.
for (int i = 0, j = 1; j < stack.depth(); i++, j++) {
if (stack[i] > stack[j]) {
return false;
}
}
return true;
}
} // namespace
SmarterStack::SmarterStack(int size) : size_(size), top_(-1) {
if (size <= 0) {
const std::length_error le("SmarterStack depth must be greater than zero.");
throw SmarterStackException(le);
}
data_ = std::unique_ptr<double[]>(new double[size]);
}
SmarterStack::SmarterStack(int size, const double *data)
: size_(size), top_(size - 1) {
data_ = std::unique_ptr<double[]>(new double[size]);
int i = 0;
while (i < size) {
data_[i] = data[i];
i++;
}
}
SmarterStack::SmarterStack(const ::std::vector<double> data)
: size_(data.size()), top_(data.size() - 1) {
data_ = std::unique_ptr<double[]>(new double[data.size()]);
// Could use a vector iterator, but still have to iterate i.
for (int i = 0; i < static_cast<int>(data.size()); i++) {
data_[i] = data.at(i);
}
}
ostream &operator<<(ostream &out, const SmarterStack &stack) {
for (int i = 0; i <= stack.top_; i++) {
out << (stack.data_[i]);
}
return out;
}
double SmarterStack::operator[](int i) {
if ((i > top_) || (i < 0)) {
range_error re("Out of range.");
throw SmarterStackException(re);
}
return data_[i];
}
double SmarterStack::operator[](int i) const {
if ((i > top_) || (i < 0)) {
range_error re("Out of range.");
throw SmarterStackException(re);
}
return data_[i];
}
bool operator==(SmarterStack &a, SmarterStack &b) {
if (a.size_ != b.size_) {
return false;
}
int i = 0;
while (i < a.size_) {
if (a[i] != b[i]) {
return false;
}
i++;
}
return true;
}
bool operator==(const SmarterStack &a, const SmarterStack &b) {
if (a.size_ != b.size_) {
return false;
}
int i = 0;
while (i < a.size_) {
if (a[i] != b[i]) {
return false;
}
i++;
}
return true;
}
// Overload the function operator to provide subsequences of the stack.
// Returns to+1 elements from 0 to to.
SmarterStack SmarterStack::operator()(int to) const {
// Last valid index in data_[] is top_.
if ((to < 0) || (to > top_)) {
range_error re("Out of range.");
throw SmarterStackException(re);
}
vector<double> temp;
for (int j = 0; j <= to; j++) {
temp.push_back(data_[j]);
}
// Implicitly convert the vector to a SmarterStack.
return temp;
}
SmarterStack SmarterStack::operator()(int from, int to) const {
// Last valid index in data_[] is top_.
if ((to > top_) || (to < from) || (from < 0)) {
range_error re("Out of range.");
throw SmarterStackException(re);
}
vector<double> temp;
for (int j = from; j <= to; j++) {
temp.push_back(data_[j]);
}
// Implicitly convert the vector to a SmarterStack.
return temp;
}
void SmarterStack::Push(double datum) {
if (full()) {
overflow_error oe("SmarterStack is full.");
throw SmarterStackException(oe);
}
// top_ is equal to the index of the the last element in the array.
data_[++top_] = datum;
}
double SmarterStack::Pop() {
if (empty()) {
underflow_error ue("SmarterStack is empty.");
throw SmarterStackException(ue);
}
// The last element that can be popped is at index 0.
return (data_[top_--]);
}
void SmarterStack::Reverse() {
SmarterStack temp(size_);
int i = 0;
while (!empty()) {
temp.Push(data_[i]);
// Pushing Pop()'ed data on temp directly results in two reversals.
Pop();
i++;
}
while (!temp.empty()) {
Push(temp.Pop());
}
}
void PrintIncreasingSubsequences(ostream &out, const SmarterStack &st) {
int from = 0, to = 0;
// depth() returns number of elements, which equals highest populated index+1.
while ((to < st.depth())) {
// A single-element stack is defined increasing.
if (!is_increasing(st(from, to))) {
out << "(";
out << st(from, to - 1);
out << "),";
// Start a new subsequence.
from = to;
}
to++;
}
// If the final element is less than its predecessor, it has not gotten
// printed.
if (from == st.depth() - 1) {
out << "(";
out << st(from, from);
out << "),";
}
}
const char *SmarterStackException::what() const noexcept {
// This code caused a use-after-free when it ran. I moved the operation to
// the constructor instead. At the time, the two variables were private and
// were in the constructor's initialization list. std::string holder =
// std::string(exception_name_) + std::string(more_); return holder.c_str();
return extended_error_.c_str();
// return "a";
}
} // namespace smarter_stack