-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathPostfix_Converter.cpp
317 lines (255 loc) · 8.59 KB
/
Postfix_Converter.cpp
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
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
// Honor Pledge:
//
// I pledge that I have neither given nor received any help
// on this assignment
#include "Postfix_Converter.h"
#include <functional>
#include <map>
#include <sstream>
//
// Lookup array for all command objects
//
struct LookupEntry {
const char* symbol;
int precedence;
std::function<Command*(Abstract_Expr_Factory&)> construct_object;
};
// clang-format off
static const LookupEntry ALL_OPERATORS[] = {
{"+", 1, [](Abstract_Expr_Factory& factory) -> Command* { return factory.construct_add_command(); }},
{"-", 1, [](Abstract_Expr_Factory& factory) -> Command* { return factory.construct_subtract_command(); }},
{"*", 2, [](Abstract_Expr_Factory& factory) -> Command* { return factory.construct_multiply_command(); }},
{"/", 2, [](Abstract_Expr_Factory& factory) -> Command* { return factory.construct_divide_command(); }},
{"%", 2, [](Abstract_Expr_Factory& factory) -> Command* { return factory.construct_modulus_command(); }}
};
// clang-format on
/**
* Search for the entry associated with a given operation symbol
*
* @param[in] token The operator to look for
* @return The associated entry
* @retval nullptr The entry is not found
*/
static const LookupEntry* find_operator(const std::string& token) {
for (size_t i = 0; i < (sizeof(ALL_OPERATORS) / sizeof(ALL_OPERATORS[0])); ++i) {
if (token == ALL_OPERATORS[i].symbol) { return ALL_OPERATORS + i; }
}
return nullptr;
}
//
// Constructor
//
Postfix_Converter::Postfix_Converter(Abstract_Expr_Factory& factory): factory(factory) {}
//
// Convert an expression to postfix
//
Queue<Command*> Postfix_Converter::convert_to_postfix(const std::string& infix_expr) {
// Reset the parser state
this->expression.clear();
this->last_token_number = false;
// Read and process all tokens
std::stringstream ss(infix_expr);
std::string token;
while (ss >> token) { this->process_single_token(token); }
// Last token needs to be a number for the expression to be valid
if (!this->last_token_number) { throw Postfix_Converter::invalid_infix_exception(); }
// Add any other elements from the stack
this->pop_remaining_operators();
return this->expression;
}
//
// Process a single token
//
void Postfix_Converter::process_single_token(const std::string& token) {
// Operands go right onto the stack
if (is_number(token)) {
this->add_number_to_expression(token);
return;
}
// Left parenthesis
if (is_left_parenthesis(token)) {
this->add_left_parenthesis(token);
return;
}
// Right parenthesis
if (is_right_parenthesis(token)) {
this->add_right_parenthesis(token);
return;
}
// Okay, treat it as a normal binary operator
if (is_operator(token)) {
this->process_operator(token);
return;
}
// Unknown symbol! Problem!
throw Postfix_Converter::invalid_operator_exception(token);
}
// Operands go right onto the stack
if (is_number(token)) {
this->add_number_to_expression(token);
return;
}
// Left parenthesis
if (is_left_parenthesis(token)) {
this->add_left_parenthesis(token);
return;
}
// Right parenthesis
if (is_right_parenthesis(token)) {
this->add_right_parenthesis(token);
return;
}
// Okay, treat it as a normal binary operator
this->process_operator(token);
}
//
// Add a number to the current expression
//
void Postfix_Converter::add_number_to_expression(const std::string& token) {
// Two numbers in a row is not allowed
if (this->last_token_number) { throw Postfix_Converter::invalid_infix_exception(); }
this->last_token_number = true;
int number = std::stoi(token);
Number_Command* num_command = factory.construct_number_command(number);
this->expression.enqueue(num_command);
}
//
// Add an operator to the current expression
//
void Postfix_Converter::add_operator_to_expression(const std::string& token) {
Command* command = get_operator_command(token);
this->expression.enqueue(command);
}
//
// Use the factory method pattern to return a command
//
Command* Postfix_Converter::get_operator_command(const std::string& token) {
// Lookup the entry in the table
const LookupEntry* lookup = find_operator(token);
if (lookup != nullptr) { return lookup->construct_object(this->factory); }
throw Postfix_Converter::invalid_operator_exception(token);
}
//
// Add a left parenthesis to the expression
//
void Postfix_Converter::add_left_parenthesis(const std::string& token) {
// Parenthesis can only come after an operator
if (this->last_token_number) { throw Postfix_Converter::invalid_infix_exception(); }
this->stack.push(token);
}
//
// Add a right parenthesis to the expression
//
void Postfix_Converter::add_right_parenthesis(const std::string& token) {
// Right parenthesis can only come after a number
if (!this->last_token_number) { throw Postfix_Converter::invalid_infix_exception(); }
this->pop_until_matching_parenthesis(token);
}
//
// Logic for processing an operator (with precedence)
//
void Postfix_Converter::process_operator(const std::string& token) {
// Operators can only come after a number
if (!this->last_token_number) { throw Postfix_Converter::invalid_infix_exception(); }
this->last_token_number = false;
// Compute operator precedence
int current_prec = get_operator_precedence(token);
int top_prec;
do {
// Always push if empty stack or last operator is left parenthesis
if (this->stack.is_empty() || is_left_parenthesis(this->stack.top())) {
this->stack.push(token);
return;
}
top_prec = get_operator_precedence(this->stack.top());
// Push immediately if the current operators has a greater precedence
if (current_prec > top_prec) {
this->stack.push(token);
return;
}
// For equal precedence, since we have left to right association, pop
// immediately
if (current_prec == top_prec) {
this->add_operator_to_expression(this->stack.pop());
this->stack.push(token);
return;
}
// Pop the element if top precedence >= current precedence, then retest the
// new top
this->add_operator_to_expression(this->stack.pop());
} while (current_prec < top_prec);
}
//
// Keep popping until the corresponding parenthesis is found
//
void Postfix_Converter::pop_until_matching_parenthesis(const std::string& right_paren) {
bool paren_found = false;
do {
// Empty Stack = no more possible parenthesis
if (this->stack.is_empty()) { throw Postfix_Converter::mismatched_parenthesis_exception(); }
// Get the top operator from the stack
std::string stack_top = stack.pop();
// Find first left parenthesis character on the stack
if (is_left_parenthesis(stack_top)) {
if (is_matching_parenthesis(stack_top, right_paren)) {
paren_found = true;
} else {
// Different types of parenthesis in wrong order
// for example: ([ )]
throw Postfix_Converter::mismatched_parenthesis_exception();
}
}
// We are not at the parenthesis yet, so add operator to the expression
if (!paren_found) { this->add_operator_to_expression(stack_top); }
} while (!paren_found);
}
//
// Pop any remaining operators from the stack
//
void Postfix_Converter::pop_remaining_operators() {
while (!this->stack.is_empty()) {
std::string stack_top = stack.pop();
// Test for any residue parenthesis
if (is_left_parenthesis(stack_top)) {
throw Postfix_Converter::mismatched_parenthesis_exception();
}
this->add_operator_to_expression(stack_top);
}
}
//
// Test if the string is a number
//
bool Postfix_Converter::is_number(const std::string& token) {
// We can use the default C++ number conversion to test if it
// is a valid number or not
char* p;
strtol(token.c_str(), &p, 10);
return *p == 0;
}
//
// Test if the input is a left parenthesis or right parenthesis
//
bool Postfix_Converter::is_left_parenthesis(const std::string& token) {
return (token == "(") || (token == "[");
}
bool Postfix_Converter::is_right_parenthesis(const std::string& token) {
return (token == ")") || (token == "]");
}
bool Postfix_Converter::is_matching_parenthesis(const std::string& left, const std::string& right) {
return ((left == "(" && right == ")") || (left == "[" && right == "]"));
}
//
// Test if the token is an operator
//
bool Postfix_Converter::is_operator(const std::string& token) {
return (find_operator(token) != nullptr);
}
//
// Get the operator precedence
//
int Postfix_Converter::get_operator_precedence(const std::string& token) {
// Lookup the entry in the table
const LookupEntry* lookup = find_operator(token);
if (lookup != nullptr) { return lookup->precedence; }
throw Postfix_Converter::invalid_operator_exception(token);
}