20212218 实验二《Python程序设计》实验报告

 # 20212218 2021-2022-2 《Python程序设计》实验二报告

 

课程:《Python程序设计》
班级: 2122
姓名: 林思凡
学号:20212218
实验教师:王志强
实验日期:2022年4月5日
必修/选修: 公选课

 

## 1.实验内容
运用老师上课讲的知识,外加上CSDN查找并潜心钻研的拓展知识,完成一个简易计算器
计算器功能如下:
- 进行单次普通二元运算,运算符包括+、-、*、/、%、//、**等
- 进行单词普通复数运算,运算符包括+、-、*、/
- 进行一串四则运算的结果输出(这是在CSDN上学的,绝对不是无条件照抄,每一步都理解了,花了我相当长时间)



## 2. 实验过程及结果
- 第一步,构造整体框架,包括普通计算、复数计算和四则运算三个方面。

 - 第二步,完成普通运算框架搭建,这一步王老师在课堂上已有所提及,我在此基础上进行了完善,提高了此程序的容错率。

比如,输入/、%、//运算符时,若除数为0则会报错。因此加入if条件判断语句避免除数为0的情况。

 - 第三步,完成复数运算框架搭建,这一步与普通运算的框架搭建方法大同小异,只不过要记住复数要引用complex()函数,且复数的形式为a+bj,而非a+bi。

 - 第四步,开始搭建四则运算框架(结果如图),这一步花费了整个程序%80的时间,在此展开叙述

 - 4-1,首先引入re模块的findall()函数,将字符串形式的四则运算式子分隔字符,储存在一个列表中。

- 4-2,将四则运算式子中小括号内的式子进行运算,然后删去括号。

  - 4-3,计算四则运算式子中的乘除运算。

 - 4-4,计算四则运算式子中的加减运算。

 - 4-5,框架搭建完毕,开始进行细化和完善。

- 4-5-1,运用if条件控制语句,避免出现两个运算符同时出现的情况,如"2+(-3)"去除括号之后会变成2+-3,这时需要去掉加号。

 - 第五步,整体框架搭建完成,开始进行整体细化。比如运用while语句给每一个部分做一个返回键和退出键。程序到这里基本完成。

 成品 完整程序如下(会托管到码云):

import re

flag1 = 1
while flag1 == 1:
    choice = input("请选择您想要的计算器类型(0为简单普通计算器、1为复数计算器、2为四则运算计算器、输入E/e退出): \n")
    if choice == "E" or choice == "e":
        break
    if choice == "0":
        flag2 = 1
        while flag2 == 1:
            op = input("请输入需要做的操作(+、-、*、/、%、//、**、输入e返回上一级、输入E退出):\n")
            if op == "e":
                break
            if op == "E":
                flag1 = 0
                break
            a = int(input("请输入操作数a:"))
            b = int(input("请输入操作数b:"))
            if op == "+":
                result = a + b
            elif op == "-":
                result = a - b
            elif op == "*":
                result = a * b
            elif op == "/":
                if b == 0:
                    print("除数不能为0!请重新输入!\n")
                    continue
                else:
                    result = a / b
            elif op == "%":
                if b == 0:
                    print("除数不能为0!请重新输入!\n")
                    continue
                else:
                    result = a % b
            elif op == "//":
                if b == 0:
                    print("除数不能为0!请重新输入!\n")
                    continue
                else:
                    result = a // b
            elif op == "**":
                result = a ** b
            else:
                print("输入有误,请重新输入!\n")
                continue
            print(a, op, b, "=", result)
        

    elif choice == "1":
        flag3 = 1
        while flag3 == 1:
            op = input("请输入需要做的操作(+、-、*、/、输入e返回上一级、输入E退出):\n")
            if op == "e":
                break
            if op == "E":
                flag1 = 0
                break

            a = complex(input("请输入第一个复数(形式为a+bj):"))
            b = complex(input("请输入第二个复数(形式为a+bj):"))
            result = 0
            if op == "+":
                result = a + b
            elif op == "-":
                result = a - b
            elif op == "*":
                result = a * b
            elif op == "/":
                result = a / b
            else:
                print("输入有误,请重新输入!\n")
            print(a, op, b, "=", result)

    if choice == '2':
        flag4 = 1
        while flag4 == 1:
            a = input("请输入一个四则运算式,我们将为您一步解出答案输入(输入e返回上一级、输入E退出):\n")
            if a == "e":
                break
            if a == "E":
                flag1 = 0
                break

            def division(n):
                list1 = re.findall('[\d\.]+|\(|\+|\-|\*|\/|\)', n)
                return list1


            def change(n, m):
                if n[m] == '-':
                    if n[m - 1] == '+':
                        n[m - 1] = '-'
                        del n[m]
                    elif n[m - 1] == '-':
                        n[m - 1] = '+'
                        del n[m]
                return n


            def senior(n):
                s_count = 0

                for i in n:
                    if i == '*':
                        if n[s_count + 1] != '-':  
                            n[s_count - 1] = float(n[s_count - 1]) * float(n[s_count + 1])
                            del n[s_count]
                            del n[s_count]
                        elif n[s_count + 1] == '-': 
                            n[s_count] = float(n[s_count - 1]) * float(n[s_count + 2])
                            n[s_count - 1] = '-'
                            del n[s_count + 1]
                        n = change(n, s_count - 1)
                        return senior(n)

                    elif i == '/':
                        if n[s_count + 1] != '-':  # 3*2
                            n[s_count - 1] = float(n[s_count - 1]) / float(n[s_count + 1])
                            del n[s_count]
                            del n[s_count]
                        elif n[s_count + 1] == '-':  # 3*-2
                            n[s_count] = float(n[s_count - 1]) / float(n[s_count + 2])
                            n[s_count - 1] = '-'
                            del n[s_count + 1]
                        n = change(n, s_count - 1)
                        return senior(n)
                    s_count = s_count + 1

                return n


            def junior(n):
                j_count = 0
                if n[0] == '-':
                    sum = 0
                else:
                    sum = float(n[0])

                for i in n:
                    if i == '-':
                        sum = sum - float(n[j_count + 1])
                    if i == '+':
                        sum = sum + float(n[j_count + 1])
                    j_count = j_count + 1

                if sum >= 0:
                    n = [str(sum)]
                else:
                    n = ['-', str(-sum)]
                return n


            def calculate(n):
                if '*' in n or '/' in n:
                    n = senior(n)
                if '+' in n or '-' in n:
                    n = junior(n)
                return n


            def no_bracket(n):
                left_bracket = 0
                count = 0
                for i in n:
                    if i == '(':
                        left_bracket = count
                    if i == ')':
                        part = n[left_bracket + 1: count]
                        resule_of_part = calculate(part)
                        n = n[: left_bracket] + resule_of_part + n[count + 1:]
                        n = change(n, left_bracket)
                        return no_bracket(n)
                    count = count + 1
                return n

            def main(n):
                n_list = division(n)
                n_list1 = no_bracket(n_list)
                answer = calculate(n_list1)
                if len(answer) == 2:
                    answer = -float(answer[1])
                else:
                    answer = float(answer[0])
                return answer
            b = main(a)
            print(f"{a} = {b}")

 

 

## 3. 实验过程中遇到的问题和解决过程
- 问题1:查阅代码时,不清楚什么是re模块
- 问题1解决方案:查阅csdn相关资料,了解re模块是一个字符匹配模块,依次拿出表达式和文本中的字符比较,如果每一个字符都能匹配,则匹配成功;一旦有匹配不成功的字符则匹配失败。如果表达式中有量词或边界,这个过程会稍微有一些不同。

 

- 问题2:查阅代码时,不了解re模块中的findall()函数
- 问题2解决方案:查阅csdn相关资料,了解findall()函数时一条用于多次查找匹配字符的函数,与re模块中的match()和search()不同,后两者只返回第一个匹配的字符。

 

- 问题3:初次使用re模块的findall()函数时,出现报错TypeError: expected string or bytes-like object
- 问题3解决方案:上CSDN查阅相关报错,发现findall语句参数只能是字符串,然后才返回一个列表;检查自己的程序时发现在参数中引入了列表,改成字符串之后便正常运行。

 

- 问题4:查阅代码时,不了解递归函数
- 问题4解决方案:查阅csdn相关资料,了解到,递归就是在运行的过程中调用自己。
构成递归需具备的条件:
1、子问题须与原始问题为同样的事,且更为简单;
2、不能无限制地调用本身,须有个出口,化简为非递归状况处理。

 

- 问题5:测试四则运算时,发现负数会自动转成正数造成计算错误
- 问题5解决方案:多次调试,发现两出问题,第一处是"或"条件判断书写存在错误;第二处是在一处计算当中忘记添加负号,改正之后解决,程序运行正常。



## 其他(感悟、思考等)

 

1. 调试真的非常非常好用!此次程序编写行数为188行。当你面对一大堆代码找不出问题所在时,不要犹豫,进行调试!通过调试,你能洞悉代码每一步的运行情况,运筹帷幄,掌握全局,揪出漏洞,及时填补!

 

2. 网络是非常好非常好的学习平台,在这里你能够获得程序员大佬们悉心传授的知识,并且受益无穷!本次计算器设计的四则运算部分,就是在我深入剖析大佬的代码之后,恍然大悟才写出来的。

 

## 参考资料

 

-  [什么是递归,通过这篇文章,让你彻底搞懂递归](https://blog.csdn.net/abcdef314159/article/details/107906395?ops_request_misc=%257B%2522request%255Fid%2522%253A%2522164912286116780269818899%2522%252C%2522scm%2522%253A%252220140713.130102334..%2522%257D&request_id=164912286116780269818899&biz_id=0&utm_medium=distribute.pc_search_result.none-task-blog-2~all~top_positive~default-2-107906395.142^v5^pc_search_result_control_group,157^v4^control&utm_term=%E9%80%92%E5%BD%92&spm=1018.2226.3001.4187)

 

-  [python——正则表达式(re模块)详解](https://blog.csdn.net/guo_qingxia/article/details/113979135?ops_request_misc=%257B%2522request%255Fid%2522%253A%2522164912120316780274154982%2522%252C%2522scm%2522%253A%252220140713.130102334..%2522%257D&request_id=164912120316780274154982&biz_id=0&utm_medium=distribute.pc_search_result.none-task-blog-2~all~top_positive~default-1-113979135.142^v5^pc_search_result_control_group,157^v4^control&utm_term=re%E6%A8%A1%E5%9D%97&spm=1018.2226.3001.4187)
 
-  [如何简单地理解Python中的if __name__ == '__main__'](https://blog.csdn.net/yjk13703623757/article/details/77918633?ops_request_misc=%257B%2522request%255Fid%2522%253A%2522164913249116780271929101%2522%252C%2522scm%2522%253A%252220140713.130102334..%2522%257D&request_id=164913249116780271929101&biz_id=0&utm_medium=distribute.pc_search_result.none-task-blog-2~all~top_positive~default-1-77918633.142^v5^pc_search_result_control_group,157^v4^control&utm_term=+if+__name__+%3D%3D+__main__%3A&spm=1018.2226.3001.4187)
 
-  [项目3:python实现一个简易计算器](https://blog.csdn.net/a971956955/article/details/81489914?ops_request_misc=&request_id=&biz_id=102&utm_term=python%E8%AE%A1%E7%AE%97%E5%99%A8&utm_medium=distribute.pc_search_result.none-task-blog-2~all~sobaiduweb~default-1-81489914.nonecase&spm=1018.2226.3001.4187)

 

posted @ 2022-04-05 16:19  特别中二的文年同学  阅读(174)  评论(0编辑  收藏  举报