-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathArithmeticExpEval.go
115 lines (92 loc) · 3.19 KB
/
ArithmeticExpEval.go
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
/*
Given a string containing a list of numbers separated by arithmetic operators, write a program to output the correct result.
Example 1:
Input: "1 + 2"
Output: 3
Example 2:
Input: "3 + 2 - 1 * 6 / 5 - 1 + 8"
Output: 10.8
Notes:
You may assume operators and operands are separated by one or more spaces.
The only operators would be +,-,*,/ and the operands would be decimal digits.
*/
package main
import (
"log"
"strconv"
"strings"
)
func main() {
tests := []string{"1 + 2", "1 * 2", "2 / 3", "22 / 7", "3 + 2 - 1", "4 / 2", "4 / 2 + 1", "4 / 2 - 1", "3 + 2 - 1 * 6 / 5 - 1 + 8"}
for _, test := range tests {
log.Printf("evaluate(%s) == %f\n", test, evaluate(test))
}
}
func evaluate(expr string) float64 {
// exprRunes = []rune(expr)
// for i := 0; i < len(exprRunes); i++ {
// if expRunes
// }
tokens := strings.Split(expr, " ")
for _, token := range tokens {
token = strings.TrimSpace(token)
if token == "*" {
// https://golang.org/pkg/strings/#SplitAfterN
// fmt.Printf("%q\n", strings.SplitAfterN("a,b,c,d,e,f", ",", 2))
// == ["a," "b,c,d,e,f"]
comps := strings.SplitAfterN(expr, "*", 2)
firstSubExp := comps[0]
secondSubExp := comps[1]
// firstSubExpRunes will be something like "subexp *" - get rid of that trailing *
firstSubExpRunes := []rune(firstSubExp)
firstSubExpRunes = firstSubExpRunes[:len(firstSubExpRunes)-1]
firstSubExp = string(firstSubExpRunes)
return evaluate(firstSubExp) * evaluate(secondSubExp)
}
if token == "/" {
comps := strings.SplitAfterN(expr, "/", 2)
firstSubExp := comps[0]
secondSubExp := comps[1]
// firstSubExpRunes will be something like "subexp *" - get rid of that trailing *
firstSubExpRunes := []rune(firstSubExp)
firstSubExpRunes = firstSubExpRunes[:len(firstSubExpRunes)-1]
firstSubExp = string(firstSubExpRunes)
return evaluate(firstSubExp) / evaluate(secondSubExp)
}
}
// no *'s or /'s found
for _, token := range tokens {
if token == "+" {
// https://golang.org/pkg/strings/#SplitAfterN
// fmt.Printf("%q\n", strings.SplitAfterN("a,b,c,d,e,f", ",", 2))
// == ["a," "b,c,d,e,f"]
comps := strings.SplitAfterN(expr, "+", 2)
firstSubExp := comps[0]
secondSubExp := comps[1]
// firstSubExpRunes will be something like "subexp *" - get rid of that trailing *
firstSubExpRunes := []rune(firstSubExp)
firstSubExpRunes = firstSubExpRunes[:len(firstSubExpRunes)-1]
firstSubExp = string(firstSubExpRunes)
return evaluate(firstSubExp) + evaluate(secondSubExp)
}
if token == "-" {
// https://golang.org/pkg/strings/#SplitAfterN
// fmt.Printf("%q\n", strings.SplitAfterN("a,b,c,d,e,f", ",", 2))
// == ["a," "b,c,d,e,f"]
comps := strings.SplitAfterN(expr, "-", 2)
firstSubExp := comps[0]
secondSubExp := comps[1]
// firstSubExpRunes will be something like "subexp *" - get rid of that trailing *
firstSubExpRunes := []rune(firstSubExp)
firstSubExpRunes = firstSubExpRunes[:len(firstSubExpRunes)-1]
firstSubExp = string(firstSubExpRunes)
return evaluate(firstSubExp) - evaluate(secondSubExp)
}
}
// no operators (/ * + -) found
exprVal, err := strconv.ParseFloat(strings.TrimSpace(expr), 64)
if err != nil {
log.Fatal(err)
}
return exprVal
}