计算器源码
基于递归和正则表达式实现的计算器,源码如下:
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
|
#!/usr/bin/env python # -*- coding:utf-8 -*- """ 该计算器思路: 1、递归寻找表达式中只含有 数字和运算符的表达式,并计算结果 2、由于整数计算会忽略小数,所有的数字都认为是浮点型操作,以此来保留小数 使用技术: 1、正则表达式 2、递归 执行流程如下: ******************** 请计算表达式: 1 - 2 * ( (60-30 +(-40.0/5) * (9-2*5/3 + 7 /3*99/4*2998 +10 * 568/14 )) - (-4*3)/ (16-3*2) ) ******************** before: ['1-2*((60-30+(-40.0/5)*(9-2*5/3+7/3*99/4*2998+10*568/14))-(-4*3)/(16-3*2))'] -40.0/5=-8.0 after: ['1-2*((60-30+-8.0*(9-2*5/3+7/3*99/4*2998+10*568/14))-(-4*3)/(16-3*2))'] ========== 上一次计算结束 ========== before: ['1-2*((60-30+-8.0*(9-2*5/3+7/3*99/4*2998+10*568/14))-(-4*3)/(16-3*2))'] 9-2*5/3+7/3*99/4*2998+10*568/14=173545.880953 after: ['1-2*((60-30+-8.0*173545.880953)-(-4*3)/(16-3*2))'] ========== 上一次计算结束 ========== before: ['1-2*((60-30+-8.0*173545.880953)-(-4*3)/(16-3*2))'] 60-30+-8.0*173545.880953=-1388337.04762 after: ['1-2*(-1388337.04762-(-4*3)/(16-3*2))'] ========== 上一次计算结束 ========== before: ['1-2*(-1388337.04762-(-4*3)/(16-3*2))'] -4*3=-12.0 after: ['1-2*(-1388337.04762--12.0/(16-3*2))'] ========== 上一次计算结束 ========== before: ['1-2*(-1388337.04762--12.0/(16-3*2))'] 16-3*2=10.0 after: ['1-2*(-1388337.04762--12.0/10.0)'] ========== 上一次计算结束 ========== before: ['1-2*(-1388337.04762--12.0/10.0)'] -1388337.04762--12.0/10.0=-1388335.84762 after: ['1-2*-1388335.84762'] ========== 上一次计算结束 ========== 我的计算结果: 2776672.69524 """ import re def compute_mul_div(arg): """ 操作乘除 :param expression:表达式 :return:计算结果 """ val = arg[ 0 ] mch = re.search( '\d+\.*\d*[\*\/]+[\+\-]?\d+\.*\d*' , val) if not mch: return content = re.search( '\d+\.*\d*[\*\/]+[\+\-]?\d+\.*\d*' , val).group() if len (content.split( '*' ))> 1 : n1, n2 = content.split( '*' ) value = float (n1) * float (n2) else : n1, n2 = content.split( '/' ) value = float (n1) / float (n2) before, after = re.split( '\d+\.*\d*[\*\/]+[\+\-]?\d+\.*\d*' , val, 1 ) new_str = "%s%s%s" % (before,value,after) arg[ 0 ] = new_str compute_mul_div(arg) def compute_add_sub(arg): """ 操作加减 :param expression:表达式 :return:计算结果 """ while True : if arg[ 0 ].__contains__( '+-' ) or arg[ 0 ].__contains__( "++" ) or arg[ 0 ].__contains__( '-+' ) or arg[ 0 ].__contains__( "--" ): arg[ 0 ] = arg[ 0 ].replace( '+-' , '-' ) arg[ 0 ] = arg[ 0 ].replace( '++' , '+' ) arg[ 0 ] = arg[ 0 ].replace( '-+' , '-' ) arg[ 0 ] = arg[ 0 ].replace( '--' , '+' ) else : break if arg[ 0 ].startswith( '-' ): arg[ 1 ] + = 1 arg[ 0 ] = arg[ 0 ].replace( '-' , '&' ) arg[ 0 ] = arg[ 0 ].replace( '+' , '-' ) arg[ 0 ] = arg[ 0 ].replace( '&' , '+' ) arg[ 0 ] = arg[ 0 ][ 1 :] val = arg[ 0 ] mch = re.search( '\d+\.*\d*[\+\-]{1}\d+\.*\d*' , val) if not mch: return content = re.search( '\d+\.*\d*[\+\-]{1}\d+\.*\d*' , val).group() if len (content.split( '+' ))> 1 : n1, n2 = content.split( '+' ) value = float (n1) + float (n2) else : n1, n2 = content.split( '-' ) value = float (n1) - float (n2) before, after = re.split( '\d+\.*\d*[\+\-]{1}\d+\.*\d*' , val, 1 ) new_str = "%s%s%s" % (before,value,after) arg[ 0 ] = new_str compute_add_sub(arg) def compute(expression): """ 操作加减乘除 :param expression:表达式 :return:计算结果 """ inp = [expression, 0 ] # 处理表达式中的乘除 compute_mul_div(inp) # 处理 compute_add_sub(inp) if divmod (inp[ 1 ], 2 )[ 1 ] = = 1 : result = float (inp[ 0 ]) result = result * - 1 else : result = float (inp[ 0 ]) return result def exec_bracket(expression): """ 递归处理括号,并计算 :param expression: 表达式 :return:最终计算结果 """ # 如果表达式中已经没有括号,则直接调用负责计算的函数,将表达式结果返回,如:2*1-82+444 if not re.search( '\(([\+\-\*\/]*\d+\.*\d*){2,}\)' , expression): final = compute(expression) return final # 获取 第一个 只含有 数字/小数 和 操作符 的括号 # 如: # ['1-2*((60-30+(-40.0/5)*(9-2*5/3+7/3*99/4*2998+10*568/14))-(-4*3)/(16-3*2))'] # 找出:(-40.0/5) content = re.search( '\(([\+\-\*\/]*\d+\.*\d*){2,}\)' , expression).group() # 分割表达式,即: # 将['1-2*((60-30+(-40.0/5)*(9-2*5/3+7/3*99/4*2998+10*568/14))-(-4*3)/(16-3*2))'] # 分割更三部分:['1-2*((60-30+( (-40.0/5) *(9-2*5/3+7/3*99/4*2998+10*568/14))-(-4*3)/(16-3*2))'] before, nothing, after = re.split( '\(([\+\-\*\/]*\d+\.*\d*){2,}\)' , expression, 1 ) print 'before:' ,expression content = content[ 1 : len (content) - 1 ] # 计算,提取的表示 (-40.0/5),并活的结果,即:-40.0/5=-8.0 ret = compute(content) print '%s=%s' % ( content, ret) # 将执行结果拼接,['1-2*((60-30+( -8.0 *(9-2*5/3+7/3*99/4*2998+10*568/14))-(-4*3)/(16-3*2))'] expression = "%s%s%s" % (before, ret, after) print 'after:' ,expression print "=" * 10 , '上一次计算结束' , "=" * 10 # 循环继续下次括号处理操作,本次携带者的是已被处理后的表达式,即: # ['1-2*((60-30+ -8.0 *(9-2*5/3+7/3*99/4*2998+10*568/14))-(-4*3)/(16-3*2))'] # 如此周而复始的操作,直到表达式中不再含有括号 return exec_bracket(expression) # 使用 __name__ 的目的: # 只有执行 python index.py 时,以下代码才执行 # 如果其他人导入该模块,以下代码不执行 if __name__ = = "__main__" : #print '*'*20,"请计算表达式:", "1 - 2 * ( (60-30 +(-40.0/5) * (9-2*5/3 + 7 /3*99/4*2998 +10 * 568/14 )) - (-4*3)/ (16-3*2) )" ,'*'*20 #inpp = '1 - 2 * ( (60-30 +(-40.0/5) * (9-2*5/3 + 7 /3*99/4*2998 +10 * 568/14 )) - (-4*3)/ (16-3*2) ) ' inpp = "1-2*-30/-12*(-20+200*-3/-200*-300-100)" #inpp = "1-5*980.0" inpp = re.sub( '\s*' ,'',inpp) # 表达式保存在列表中 result = exec_bracket(inpp) print result |