-
Notifications
You must be signed in to change notification settings - Fork 11
/
Copy pathcalc.py
139 lines (107 loc) · 4.05 KB
/
calc.py
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
import operator
ops = ['+', '-', '*', '/']
op_mask = {
'+': operator.add,
'-': operator.sub,
'*': operator.mul,
'/': operator.truediv
}
def calc(op, item1=0, item2=0):
return op_mask[op](int(item1), int(item2))
def is_operator(ch):
return ch in ops
def next_operator(string):
for c in string:
if is_operator(c):
return c
return False
def previous_operator(string):
reversed_string = string[::-1]
return next_operator(reversed_string)
def is_minus_sign(string):
return string.count('-') == 1 and string.index('-') == 0
def still_operators(string):
for operation in ops:
if string.count(operation) > 0 and not (operation == '-' and is_minus_sign(string)):
return True
return False
def get_terms(string, position):
# -- Split the string in two parts by the operator --
string_item1 = string[:position]
string_item2 = string[position + 1:]
# -- Looking for size of the first term --
previous_operation = previous_operator(string_item1)
if previous_operation:
item1 = string_item1.split(previous_operation)[-1]
# -- If previous operator is "-" we assume is the sign of the number and we add it --
if previous_operation == '-' and is_minus_sign(string_item1):
item1 = '-' + item1
else:
item1 = string_item1
# -- Looking for size of the second term --
next_operation = next_operator(string_item2)
if next_operation:
item2 = string_item2[0:string_item2.index(next_operation)]
else:
item2 = string_item2
return (item1, item2, next_operation)
def normalize(line):
result = ''
line = ''.join(line.split())
# -- If line starts with (-) or (+) we keep this for the result --
if line[0] in ['+', '-']:
result = line[0]
line = line[1:]
subline = line
list_operations = []
while next_operator(subline):
operation = next_operator(subline)
position = subline.index(operation)
subline = subline[position + 1:]
list_operations.append(operation)
subline = line
for operation in ops:
subline = subline.replace(operation, '#')
list_words = subline.split('#')
for position in range(len(list_words)):
word = list_words[position].strip()
try:
int(word)
except ValueError:
print('Unable to parse word: ' + word)
result += word
if position < len(list_operations):
result += list_operations[position]
return result
def run(string):
string = normalize(string)
assert string[0] not in ['*', '/'], 'Error: Operator(* or /) can\'t be at the beginning of the instruction'
# -- Until we don't find more operators in the string we keep looping --
result = 0
while still_operators(string):
operation = next_operator(string)
position = string.index(operation)
# -- Avoiding (-) symbol at the beginning --
if operation == '-' and position == 0:
substring = string[1:]
operation = next_operator(substring)
if not operation:
break
subposition = substring.index(operation)
position += subposition + 1
# -- Getting terms involve in the operation --
(item1, item2, next_operation) = get_terms(string, position)
string = string[position + len(item2) + 1:]
# -- If current operation is + or - and next operation is / or *
# -- the second one has preference over the first one. Otherwise we just calculate --
if next_operation and operation in ['+', '-'] and next_operation in ['/', '*']:
(_, item3, _) = get_terms(string, 0)
item2 = calc(next_operation, item2, item3)
string = str(item1) + operation + str(item2) + string[len(item3) + 1:]
else:
result = calc(operation, item1, item2)
string = str(result) + string
return int(result)
if __name__ == '__main__':
inp = input('Enter your calculation request:\n')
print(run(inp))