python正则表达式之作业计算器
admin
2023-07-31 02:30:45
0

作业:计算器开发

实现加减乘除及拓号优先级解析
用户输入 1 – 2 * ( (60-30 +(-40/5) * (9-2*5/3 + 7 /3*99/4*2998 +10 * 568/14 )) – (-4*3)/ (16-3*2) )等类似公式后,必须自己解析里面的(),+,-,*,/符号和公式,运算后得出结果,结果必须与真实的计算器所得出的结果一致
一、说明:

有一点bug就是不能计算幂次方,如:\’6**6\’会报错

该计算器思路:
1、没用使用递归,先找出并计算所有括号里的公式,再计算乘除和加减
2、所有的数字都认为是浮点型操作,以此来保留小数
使用技术:
1、正则表达式
2、tkinter

二、流程图:

 

 三、代码如下:

#!/usr/bin/env python3
#antuor:Alan
 
import re
from functools import reduce
from tkinter import *
 
 
 
\'\'\'处理特殊-号运算\'\'\'
def minus_operation(expresstion):
  minus_operators = re.split(\"-\",expresstion)
  calc_list = re.findall(\"[0-9]\",expresstion)
  if minus_operators[0] ==\"\":
    calc_list[0] = \'-%s\' % calc_list[0]
  res = reduce(lambda x,y:float(x)-float(y),calc_list)
  print(\">>>>>>>>>>>>>>减号[%s]运算结果:\" % expresstion,res)
  return res
 
\'\'\'reduce()对sequence连续使用function, 如果不给出initial, 则第一次调用传递sequence的两个元素,
以后把前一次调用的结果和sequence的下一个元素传递给function\'\'\'
 
 
 
 
\'\'\'处理双运算符号\'\'\'
def del_duplicates(expresstion):
  expresstion = expresstion.replace(\"++\",\"+\")
  expresstion = expresstion.replace(\"--\",\"-\")
  expresstion = expresstion.replace(\"+-\",\"-\")
  expresstion = expresstion.replace(\"--\",\"+\")
  expresstion = expresstion.replace(\'- -\',\"+\")
  e
  return expresstion
 
\'\'\'*/运算函数\'\'\'
def mutiply_dividend(expresstion):
  calc_list = re.split(\"[*/]\",expresstion)     #用* or /分割公式
  operators = re.findall(\"[*/]\",expresstion)    #找出所有*和/号
  res = None
  for index,i in enumerate(calc_list):
    if res:
      if operators[index-1] ==\'*\':
        res *= float(i)
      elif operators[index-1] ==\'/\':
        res /=float(i)
    else :
      res = float(i)
  procession0 = \"[%s]运算结果=\" % expresstion,res
  final_result.insert(END,procession0)    #插入窗体
  print(procession0)
  return res
 
 
 
\'\'\'处理运算符号顺序混乱情况\'\'\'
def special_features(plus_and_minus_operators,multiply_and_dividend):
 
  for index,i in enumerate(multiply_and_dividend):
    i = i.strip()
    if i.endswith(\"*\") or i.endswith(\"/\"):
      multiply_and_dividend[index] = multiply_and_dividend[index] + plus_and_minus_operators[index] + multiply_and_dividend[index+1]
      del multiply_and_dividend[index+1]
      del plus_and_minus_operators[index]
  return plus_and_minus_operators,multiply_and_dividend
 
 
 
def minus_special(operator_list,calc_list):
  for index,i in enumerate(calc_list):
    if i ==\'\':
      calc_list[index+1] = i + calc_list[index+1].strip()
 
 
 
\'\'\'运算除了()的公式+-*/\'\'\'
def figure_up(expresstion):
  expresstion = expresstion.strip(\"()\")    #去掉外面括号
  expresstion = del_duplicates(expresstion)  #去掉重复+-号
  plus_and_minus_operators = re.findall(\"[+-]\",expresstion)
  multiply_and_dividend = re.split(\"[+-]\",expresstion)
  if len(multiply_and_dividend[0].strip()) ==0:
    multiply_and_dividend[1] = plus_and_minus_operators[0] + multiply_and_dividend[1]
    del multiply_and_dividend[0]
    del plus_and_minus_operators[0]
 
  plus_and_minus_operators,multiply_and_dividend = special_features(plus_and_minus_operators,multiply_and_dividend)
  for index,i in enumerate(multiply_and_dividend):
    if re.search(\"[*/]\",i):
      sub_res = mutiply_dividend(i)
      multiply_and_dividend[index] = sub_res
 
  print(multiply_and_dividend,plus_and_minus_operators)  #计算
  final_res = None
  for index,item in enumerate(multiply_and_dividend):
    if final_res:
      if plus_and_minus_operators[index-1] == \'+\':
        final_res += float(item)
      elif plus_and_minus_operators[index-1] == \'-\':
        final_res -= float(item)
    else:
      final_res = float(item)
  procession = \'[%s]计算结果:\' % expresstion,final_res
  final_result.insert(END,procession)    #插入窗体
  print(procession)
  return final_res
 
\"\"\"主函数:运算逻辑:先计算拓号里的值,算出来后再算乘除,再算加减\"\"\"
def calculate():
  expresstion = expresstions.get()         #获取输入框值
  flage = True
  calculate_res = None               #初始化计算结果为None
  while flage:
    m = re.search(\"\\([^()]*\\)\",expresstion)    #先找最里层的()
    # pattern = re.compile(r\"\\([^()]*\\)\")
    # m = pattern.match(expresstion)
    if m:
      sub_res = figure_up(m.group())      #运算()里的公式
      expresstion = expresstion.replace(m.group(),str(sub_res))  #运算完毕把结果替换掉公式
    else:
      print(\'---------------括号已经计算完毕--------------\')
      procession1 = \"最终计算结果:\",figure_up(expresstion)
      final_result.insert(END,procession1)   #插入窗体
      print(\'\\033[31m最终计算结果:\',figure_up(expresstion))
 
      flage = False
 
 
if __name__==\"__main__\":
  # res = calculate(\"1 - 2 * ( (60-30 +(-40/5) * (9-2*5/3 + 7 /3*99/4*2998 +10 * 568/14 )) - (-4*3)/ (16-3*2) )\")
  window = Tk()         ###创建窗体
  window.title(\'计算器\')    ###命名窗体
  frame1 = Frame(window)    ###框架1
  frame1.pack()        ###放置
  frame2 = Frame(window)    ###框架2
  frame2.pack()        ###放置
  lable = Label(frame1,text = \"请输入公式:\")  ###文字标签
  lable.pack()
  expresstions = StringVar()  ###输入框属性,字符串
  entryname = Entry(frame1,textvariable = expresstions) ###文本输入框
  bt_get_expresstions = Button(frame1,text = \"提交\",command = calculate) ###按钮挂件
  bt_get_expresstions.pack()
  entryname.pack()
  lable.grid(row =1,column =1)  ###位置
  entryname.grid(row=1,column =2)
  bt_get_expresstions.grid(row =1,column =3)
  final_result = Text(frame2)  ###计算结果显示框
  final_result.tag_config(\"here\", background=\"yellow\", foreground=\"blue\")
  final_result.pack()
  window.mainloop()       ###事件循环

以上就是本文的全部内容,希望对大家的学习有所帮助。

相关内容

热门资讯

500 行 Python 代码... 语法分析器描述了一个句子的语法结构,用来帮助其他的应用进行推理。自然语言引入了很多意外的歧义,以我们...
定时清理删除C:\Progra... C:\Program Files (x86)下面很多scoped_dir开头的文件夹 写个批处理 定...
65536是2的几次方 计算2... 65536是2的16次方:65536=2⁶ 65536是256的2次方:65536=256 6553...
Mobi、epub格式电子书如... 在wps里全局设置里有一个文件关联,打开,勾选电子书文件选项就可以了。
scoped_dir32_70... 一台虚拟机C盘总是莫名奇妙的空间用完,导致很多软件没法再运行。经过仔细检查发现是C:\Program...
小程序支付时提示:appid和... [Q]小程序支付时提示:appid和mch_id不匹配 [A]小程序和微信支付没有进行关联,访问“小...
pycparser 是一个用... `pycparser` 是一个用 Python 编写的 C 语言解析器。它可以用来解析 C 代码并构...
微信小程序使用slider实现... 众所周知哈,微信小程序里面的音频播放是没有进度条的,但最近有个项目呢,客户要求音频要有进度条控制,所...
python查找阿姆斯特朗数 题目解释 如果一个n位正整数等于其各位数字的n次方之和,则称该数为阿姆斯特朗数。 例如1^3 + 5...
Apache Doris 2.... 亲爱的社区小伙伴们,我们很高兴地向大家宣布,Apache Doris 2.0.0 版本已于...