-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathcalculate.js
122 lines (114 loc) · 2.88 KB
/
calculate.js
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
/**
*
* 使用js实现四则运算
*
**/
var A = '((112 + 2) * (32 + (43 + 45 - 46) * 12))';
function is_op(val) {
var op_string = '+-*/()';
return op_string.indexOf(val) > -1;
}
function init_expression(expression) {
var expression = expression.replace(/\s/g, ''),
input_stack = [];
input_stack.push(expression[0]);
for (var i = 1; l = expression.length, i<l; i++) {
if (is_op(expression[i]) || is_op(input_stack.slice(-1))) {
input_stack.push(expression[i]);
} else {
input_stack.push(input_stack.pop()+expression[i]);
}
}
return input_stack;
}
function op_level (op) {
if (op == '+' || op == '-') {
return 0;
}
if (op == '*' || op == '/') {
return 1;
}
if (op == '(') {
return 3;
}
if (op == ')') {
return 4;
}
}
function RPN (input_stack) {
var out_stack = [], op_stack = [], match = false, tmp_op;
while (input_stack.length > 0 ) {
var sign = input_stack.shift();
if (!is_op(sign)) {
out_stack.push(sign);
} else if (op_level(sign) == 4) {
match = false;
while (op_stack.length > 0 ) {
tmp_op = op_stack.pop();
if ( tmp_op == '(') {
match = true;
break;
} else {
out_stack.push(tmp_op);
}
}
if (match == false) {
return 'lack left';
}
} else {
while ( op_stack.length > 0 && op_stack.slice(-1) != '(' && op_level(sign) <= op_level(op_stack.slice(-1))) {
out_stack.push(op_stack.pop());
}
op_stack.push(sign);
}
}
while (op_stack.length > 0 ){
var tmp_op = op_stack.pop();
if (tmp_op != '(') {
out_stack.push(tmp_op);
} else {
return 'lack right';
}
}
return out_stack;
}
function cal(expression) {
var i, j,
RPN_exp = [],
ans;
while (expression.length > 0) {
var sign = expression.shift();
if (!is_op(sign)) {
RPN_exp.push(sign);
} else {
j = parseFloat(RPN_exp.pop());
i = parseFloat(RPN_exp.pop());
RPN_exp.push(cal_two(i, j, sign));
}
}
return RPN_exp[0];
}
function cal_two(i, j, sign) {
switch (sign) {
case '+':
return i + j;
break;
case '-':
return i - j;
break;
case '*':
return i * j;
break;
case '/':
return i / j;
break;
default:
return false;
}
}
var expression = RPN(init_expression(A))
if (expression == 'lack left' || expression == 'lack right') {
console.log(expression);
} else {
console.log(cal(expression));
}