-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathsmarter_stack.hh
122 lines (109 loc) · 4.13 KB
/
smarter_stack.hh
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
#include <cerrno>
#include <exception>
#include <iostream>
#include <memory>
#include <stdexcept>
#include <typeinfo>
#include <vector>
namespace smarter_stack {
// Provide an error message for the indicated failure.
class SmarterStackException : public std::exception {
public:
// Setting extended_error_ in the initialization list caused a use-after-free
// error in the what() method.
SmarterStackException(const std::exception &e) : std::exception(e) {
extended_error_ =
std::string(typeid(e).name()) + ": " + std::string(e.what());
}
const char *what() const noexcept;
~SmarterStackException() {}
private:
std::string extended_error_;
};
class SmarterStack {
public:
SmarterStack() : size_(0), top_(-1){};
// https://en.cppreference.com/w/cpp/memory/new/operator_new#Class-specific_overloads
// The example code shows the replacement new called only from
// non-class-member main() functions. I tried putting
// clang-format off
// SmarterStack *st = new(true) SmarterStack(int);
// clang-format on
// but doing so does not call the SmarterStack constructor and therefore
// does not improve test coverage.
/* static void *operator new(size_t sz, bool b = false) throw() {
if (b) {
std::cerr << "Returning nullptr instead of " << sz << " allocation."
<< std::endl;
return nullptr;
} else {
// The following line compiles with C++17 only.
return ::operator new(sz);
// return std::malloc(sz);
}
}
SmarterStack(int size) : size_(size), top_(-1) {
//
https://en.cppreference.com/w/cpp/memory/new/operator_new#Class-specific_overloads
// SmarterStack(int size, bool fail = false) : size_(size), top_(-1) {
std::cerr << "Size ctor" << std::endl;
if (size <= 0) {
const std::length_error le(
"SmarterStack depth must be greater than zero.");
throw SmarterStackException(le);
}
// even with C++17:
// clang-format off
// error: no matching function for call to ‘operator new [](sizetype,
bool&)’
// data_ = new(fail) double[size];
// data_ = new(double[size], fail);
// error: expected type-specifier before ‘)’ token
// data_ = static_cast<double *>(new (size * sizeof(double), fail));
// clang-format on
if (nullptr == data_) {
throw std::bad_alloc();
}
} */
// Make explicit to avoid confusion with subsequence operator.
explicit SmarterStack(int size);
// Allow for stack to be partially full by passing size as a separate
// parameter.
SmarterStack(int size, const double *data);
SmarterStack(const ::std::vector<double> data);
SmarterStack(const SmarterStack &st) = delete;
SmarterStack(SmarterStack &&st) = default;
void Push(double datum);
double Pop();
void Reverse();
friend ::std::ostream &operator<<(::std::ostream &out,
const SmarterStack &st);
friend bool operator==(SmarterStack &a, SmarterStack &b);
friend bool operator==(const SmarterStack &a, const SmarterStack &b);
bool full() const { return (top_ + 1 == size_); }
bool empty() const { return (-1 == top_); }
void clear() { top_ = -1; }
// depth() returns the number of populated elements, which may be < size_.
int depth() const { return top_ + 1; }
double operator[](int i);
double operator[](int i) const;
// Return a subsequence of the stack starting with from of length
// ((top_-from)+1).
SmarterStack operator()(int from) const;
// Return a subsequence of the stack starting with from and ending with to, of
// length ((to - from) + 1).
SmarterStack operator()(int from, int to) const;
private:
// Number of elements in data_.
int size_;
// Index of topmost element in data_ == size_-1.
int top_;
std::unique_ptr<double[]> data_;
};
::std::ostream &operator<<(::std::ostream &out, const SmarterStack &st);
bool operator==(SmarterStack &a, SmarterStack &b);
bool operator==(const SmarterStack &a, const SmarterStack &b);
// Print into an output stream all the monotonically increasing subsequences
// found in the stack.
void PrintIncreasingSubsequences(::std::ostream &out, const SmarterStack &st);
} // namespace smarter_stack