Github项目地址:

https://github.com/JtvDeemo/elementary-arithmetic

PSP

PSP2.1 Personal Software Process Stages 预估耗时(分钟) 实际耗时(分钟)
Planning 计划 10 10
· Estimate · 估计这个任务需要多少时间 1440 920
Development 开发 700 200
· Analysis · 需求分析 (包括学习新技术) 180 240
· Design Spec · 生成设计文档 5 5
· Design Review · 设计复审 (和同事审核设计文档) 10 15
· Coding Standard · 代码规范 (为目前的开发制定合适的规范) 5 5
· Design · 具体设计 40 60
· Coding · 具体编码 300 380
· Code Review · 代码复审 30 30
· Test · 测试(自我测试,修改代码,提交修改) 30 30
Reporting 报告 120 120
· Test Report · 测试报告+博客 120 120
· Size Measurement · 计算工作量 10 10
· Postmortem & Process Improvement Plan · 事后总结, 并提出过程改进计划 40 50
合计 3040 2195

题目要求:

  • 能自动生成小学四则运算题目
  • 除了整数外,还要支持真分数的四则运算

    除了以上的基本需求,还有
  • 生成的算式长度随机
  • 能处理分数的运算,结果能用分数(字符串类型)表示出来,能与用户输入相对应

解题思路:

  • 定义一个函数用于随机生成随机长度的算式
  • 把字符串型的算式转换为逆波兰式(RPN,也称后缀表达式)
  • 再把后缀表达式利用栈结构计算出结果
  • 最后再与用户的输入做比较

重点难点:

  1. 分数的表示与计算
  2. 后缀表达式的生成和计算
  3. 结果为负数的情况

如何解决:

  1. Python的分数计算可以用 fractions 库
  2. https://blog.csdn.net/qq_36763635/article/details/72627601 这里介绍了如何将中缀表达式转换为后缀表达式(RPN)
  3. https://blog.csdn.net/yangquanhui1991/article/details/52187375 图解后缀表达式的计算过程
  4. 对于结果为负数的情况,只能生成算式之后验算一遍,若为负数的情况,重新生成一遍(可能是个人水平有限)

设计实现:

具体程序设计:

成员变量

成员名 类型 功能
op list 存放运算符
quest str 存放算式
lens int 2到9的随机长度
teop str 存放当前运算符
tstr str 存放当前算式
tint int 存放当前运算数

成员函数

函数名 输入 输出 依赖函数 功能
get_string 字符串 随机生成一个算式
get_ans str 返回布尔类型 get_string 将用户输入与正确答案比较
to_rpn str 返回后缀表达式 get_ans 将随机生成的算式转换为RPN
this_bigger str,str 返回布尔表达式 冇啊 比较两个运算符的优先级
slove_rpn str 返回计算结果 get_ans 将后缀表达式计算出来

核心代码:

  1. #随机生成一个算式
  2. def get_string(self):
  3. self.lens = random.randint(2, 9)
  4. self.teop = ''
  5. self.tstr = []
  6. for i in range(self.lens):
  7. if self.teop == '÷':
  8. self.tint = random.randint(1, 8)
  9. self.teop = random.choice(self.op)
  10. elif self.teop == '/':
  11. self.tint = random.randint(self.tint+1, 9)
  12. self.teop = random.choice(self.op[:-1])
  13. else:
  14. self.tint = random.randint(0, 8)
  15. self.teop = random.choice(self.op)
  16. self.tstr.append(str(self.tint))
  17. self.tstr.append(self.teop)
  18. self.tstr[-1] = '='
  19. self.tstr = ''.join(self.tstr)
  20. self.quest = self.tstr
  21. return self.tstr
  1. #将随机生成的算式转换为RPN
  2. def to_rpn(self, ques): #Reverse Polish notation
  3. self.stack = []
  4. s = ''
  5. for x in ques:
  6. if x != '+' and x != '-' and x != '×' and x != '÷' and x != '/':
  7. s += x #若为数字,直接输出
  8. else: # 若为运算符,进栈
  9. if not self.stack: #栈空
  10. self.stack.append(x)
  11. else:
  12. if self.this_bigger(x, self.stack[-1]): #运算级高于栈顶元素
  13. self.stack.append(x) #直接进栈
  14. else:
  15. while self.stack:
  16. if self.this_bigger(x, self.stack[-1]):
  17. break
  18. s += self.stack.pop()
  19. self.stack.append(x)
  20. while self.stack:
  21. s += self.stack.pop()
  22. # print('在to_rpn函数中,rpn:',s)
  23. return s
  1. #将后缀表达式计算出来
  2. def slove_rpn(self, rpn):
  3. #print('进入slove_rpn函数:')
  4. self.stack1 = [] #用于保存运算数
  5. for x in rpn:
  6. if x != '+' and x != '-' and x != '×' and x != '÷' and x != '/':
  7. self.stack1.append(int(x))
  8. elif x == '+':
  9. second = self.stack1.pop()
  10. first = self.stack1.pop()
  11. self.stack1.append(first + second)
  12. elif x == '-':
  13. second = self.stack1.pop()
  14. first = self.stack1.pop()
  15. self.stack1.append(first - second)
  16. elif x == '×':
  17. second = self.stack1.pop()
  18. first = self.stack1.pop()
  19. self.stack1.append(first * second)
  20. elif x == '÷':
  21. second = self.stack1.pop()
  22. first = self.stack1.pop()
  23. self.stack1.append(Fraction(first, second))
  24. elif x == '/':
  25. second = self.stack1.pop()
  26. first = self.stack1.pop()
  27. self.stack1.append(Fraction(first, second))
  28. resault = self.stack1[0]
  29. if resault >= 0:
  30. #print('--------------题结束----------------')
  31. return resault
  32. elif resault < 0:
  33. s = self.get_string()
  34. rpn = self.to_rpn(s[:-1])
  35. return self.slove_rpn(rpn)

运行效果:

单元测试

《构建之法》第二章中详细提及了好的单元测试的标准。

  • 单元测试应该在最基本的功能/参数上检验程序的正确性。
  • 单元测试必须由最熟悉代码的人来写。
  • 单元测试过后,机器状态保持不变。
  • 单元测试要快。
  • 单元测试应该产生可重复、一致的结果。
  • 独立性-单元测试的运行/通过/失败不依赖于别的测试,可以人为构造数据,以保持测试的独立性。
  • 单元测试应该覆盖所有代码路径。

效能分析图

Pycharm中运行profiler,次数为10W次

可以看到所用的时间长度为16.04s

软工第五次作业——Python效能分析之四则运算生成器的更多相关文章

  1. 第五次作业——python效能分析与几个问题(个人作业)

    第五次作业--效能分析与几个问题(个人作业) 前言 阅读了大家对于本课程的目标和规划之后,想必很多同学都跃跃欲试,迫不及待想要提高自身实践能力,那么就从第一个个人项目开始吧,题目要求见下. 阅读 阅读 ...

  2. FZU软工第五次作业-词组频率分析

    目录 00.前言: 01.分工: 02.PSP表格: 03.解题思路描述与设计实现说明: 解题思路简述: 关键代码 04.附加题设计与展示: 设计的创意独到之处 实现思路 实现成果展示 05.关键代码 ...

  3. 2020BUAA软工个人博客作业-软件案例分析

    2020BUAA软工个人博客作业-软件案例分析 17373010 杜博玮 项目 内容 这个作业属于哪个课程 2020春季计算机学院软件工程(罗杰 任健) 这个作业的要求在哪里 个人博客作业-软件案例分 ...

  4. FZU软工第十一次作业-软件产品案例分析

    目录 前言: 第一部分.调研,评测: 1.1.初次感觉: 1.2.企业号bug: 1.3.你觉得为什么这个产品组的人没有发现这些bug: 1.4.假设你们团队需要开发这套系统,需要注意哪些方面: 2. ...

  5. 福大软工 · 第十一次作业 - Alpha 事后诸葛亮(团队)

    福大软工·第十一次作业-Alpha事后诸葛亮 组长博客链接 本次作业博客链接 项目Postmortem 模板 设想和目标 我们的软件要解决什么问题?是否定义得很清楚?是否对典型用户和典型场景有清晰的描 ...

  6. 福大软工·第十一次作业-Alpha事后诸葛亮

    福大软工·第十一次作业-Alpha事后诸葛亮 组长博客链接 本次作业博客链接 项目Postmortem 模板 设想和目标 我们的软件要解决什么问题?是否定义得很清楚?是否对典型用户和典型场景有清晰的描 ...

  7. 软工网络15个人作业4——alpha阶段个人总结

    软工网络15个人作业4--alpha阶段个人总结 一.个人总结 用自我评价表:http://www.cnblogs.com/xinz/p/3852177.html 总结Alpha冲刺过程. 由于直接用 ...

  8. 软工 · 第十一次作业 - Alpha 事后诸葛亮(团队)

    软工 · 第十一次作业 - Alpha 事后诸葛亮(团队) 组长本次作业链接 现代软件工程 项目Postmortem 设想和目标 1.我们的软件要解决什么问题?是否定义得很清楚?是否对典型用户和典型场 ...

  9. 【软工】个人项目作业——个人软件流程(PSP)

    [软工]个人项目作业--个人软件流程(PSP) 项目 内容 班级:北航2020春软件工程 006班(罗杰.任健 周五) 博客园班级博客 作业:设计程序求几何对象的交点集合 个人项目作业 个人课程目标 ...

随机推荐

  1. ECMAScript 6 入门简介

    ECMAScript 6.0(以下简称ES6)是JavaScript语言的下一代标准,已经在2015年6月正式发布了.它的目标,是使得JavaScript语言可以用来编写复杂的大型应用程序,成为企业级 ...

  2. 8、Preferences

    (官网:www.libgdx.cn) Preferences是保存简单数据的一种很好的方式,比如用户设置,游戏状态等.Preferences原理像hash map,使用字符串作为键值,多种类型作为值. ...

  3. 19_Android中图片处理原理篇,关于人脸识别网站,图片加载到内存,图片缩放,图片翻转倒置,网上撕衣服游戏案例编写

    1加载图片到内存 (1).数码相机照片特别是大于3m以上的,内存吃不消,会报OutOfMemoryError,若是想只显示原图片的1/8,可以通过BitmapFactory.Options来实现,具体 ...

  4. Android ViewManger解析 从ViewRoot 源码分析invalidate

    转载请标明出处:http://blog.csdn.net/sk719887916/article/details/48443429,作者:skay     通过学习了AndroidUI之绘图机基础知道 ...

  5. java中log的应用

    log的简单应用 备忘 加入jar包commons-logging-1.1.jar log4j.properties 如下(就放在src根目录底下 名字和位置都不要变) #OFF.FATAL.ERRO ...

  6. 【一天一道LeetCode】#29. Divide Two Integers

    一天一道LeetCode系列 (一)题目 Divide two integers without using multiplication, division and mod operator. If ...

  7. 《java入门第一季》之面向对象面试题

    1:方法重写和方法重载的区别?方法重载能改变返回值类型吗? 方法重写: 在子类中,出现和父类中一模一样的方法声明的现象. 方法重载: 同一个类中,出现的方法名相同,参数列表不同的现象. 方法重载能改变 ...

  8. jsp中的tag与tld

    转载自: http://www.cnblogs.com/fanzi2009/archive/2010/04/08/1707888.html 在jsp文件中,可以引用tag和tld文件.  1.对于ta ...

  9. Rest api简介

    理解和使用内容协商 我们的开发者在发送一个 REST API 请求的同时,根据应用场景,针对相同的资源,可能会期待不同的返回形式. 比如,我希望根据用户客户端语言,同一个资源的内容可以返回不同的语言. ...

  10. 【Java编程】随机数的不重复选择

    随机数的不重复选择就是从n个数中随机选取m(m<n)个数.在本文中,我们用Java来实现.因此我们先介绍Java的相关知识. 在Java中,Java.util.Set接口和Java.util.L ...