2016011998+sw
Coding.net原码仓库地址:https://git.coding.net/laolaf/sw.git
目录
1 需求分析
2 功能设计
3 设计实现
4 算法详解
5 测试运行
6 满意代码
1 需求分析
用户分析:本软件的用户是小学生,因此,要对数据和算式进行限定。
限定条件总结如下:
1 数据是不超过1000的正整数;
2 运算符在3到5个之间;
3 运算过程中不能产生负数或分数(基本功能中)。
本软件具有以下功能:
1 接受参数n后可自动生成n道至少包含2种运算符的练习题,且不出现非整数或负数。
2 生成练习题后,学号与生成的n道练习题及其对应的正确答案输出到文件“result.txt”中。
3 在拓展类进行复杂真分数运算。
2 功能设计:
基本功能:随机生成只有正整数的四则运算算式和答案。
扩展功能:a 支持有括号的运算式,包括出题与求解正确答案。算式中存在的括号必须大于2个,且不得超过运算符的个数。
b 扩展程序功能支持真分数的出题与运算,例如:1/6 + 1/8 + 2/3= 23/24。实现本功能时,支持运算时分数的自动化简。
模块设计:
1 Cal方法:用于转化字符串并计算得到的算式的值,支持括号运算。
2 Create方法:用于生成符合要求的算式的字符串形式。
3 Add方法:支持真分数运算的模块 。
4 main方法:输出
3 设计实现:
本软件的类:Main。主要用到了2个方法。可完成以下功能:
- 基本四则运算:功能由Cal和Create完成,由Create产生合法的(运算中不出现负数和分数)的算式,再利用Cal计算。
- 加括号的四则运算:没做。
方法:
1 Cal(String str):实现计算一个多运算符的算式。参数str即是要进行运算的字符串。在算法详解中会提到。
2 CreateQuestion(int p):实现创造一个合法的算式。参数p是运算符号的个数。
它的主体:
public static String CreateQuestion(int p) {// 产生1个包含p个运算符的四则运算的方法
String re = "";//result char oper[]={'+','-','*','÷'};//operator
int Num[]=new int[30];//存储算式中的运算数
char Op[]=new char[30];//xxxxxx 运算符 for(int he=p;he>0;he--){
Op[he]=oper[(int) (Math.random()*4)];
} for(int he=p+1;he>0;he--){
Num[he]=(int) (Math.random()*100);
} //谢绝野蛮连成连÷
for(int e=p;e>0;e--){
if(Op[e]=='*'||Op[e]=='÷')
while(Op[e]!='*'&&Op[e]!='÷')
Op[1+e]=oper[(int) (Math.random()*4)];
} //合法性问题:1 负数 2 分数.
//先对乘除处理,处理之后,归纳为简单加减运算' /* 解决分数问题 */
for(int e=p;e>0;e--){
if(Op[e]=='÷') Num[e-1]*=Num[e+1]; } /*控制fu数问题,先计算已经产生的字符串,控制减数永远比被减数小,同时注意减数不能小于o*/
for(int e=0;e<p;e++){
if(Op[e] == '-')
{
String Caled= "" + Num[0];
for(int c=0; c<p; c++)
Caled= Caled + Op[c] + Op[c+1];
int existed = Cal(Caled);
Num[e+1] = existed-(int) (Math.random()*existed);
}; } re=re+Num[0];
for(int pp=0;pp<p;pp++)
re=re+Op[pp]+Num[pp+1];
return re;//返回一个合法的算式 }
4 算法详解
用到的主要算法是逆波兰表达式和调度场算法---用来计算字符串状态下算式的值。
参考了链接:https://liam0205.me/2016/12/14/Shunting-Yard-Algorithm/]
这里讲得很清楚了:
简单地说,就是先将要输入的表达式存入list中,然后,将表达式变为逆波兰表达式压入栈内,通过正则判断字符串是否为数字,是,输出到集合里,不是,则进行优先级的判断,括号的级别最高,其次是乘除,然后是加减。
部分代码:
List<String> list = (List<String>) new ArrayList<>();
Collections.addAll(list, expression); Stack<String> stack = new Stack<>(); List<String> RPNList = new ArrayList<>(); String numRegex = "^-?[0-9]+$"; for(String s : list) {
if(s.matches(numRegex)) { RPNList.add(s);
}else if("*".equals(s) || "/".equals(s) || "+".equals(s) || "-".equals(s)) { if(stack.isEmpty()) {
stack.push(s);
}else{
int currLevel = getLevel(s);
String topStr = stack.peek();
int topLevel = getLevel(topStr);
if (currLevel <= topLevel) {
// 栈顶元素依次出栈并输出
while(!stack.isEmpty()) {
// 遇到左括号退出
String peek = stack.peek();
if("(".equals(peek)) {
break;
}
RPNList.add(stack.pop());
}
}
stack.push(s);
}
}else if("(".equals(s)) {
stack.push(s);
}else if(")".equals(s)) {
// 栈顶元素依次出栈并输出
int length = stack.size();
for(int i = 0; i < length; i++) {
String pop = stack.pop();
// 遇到左括号退出 并丢弃“”"("
if("(".equals(pop)) {
break;
}
RPNList.add(pop);
}
}
}
// 如果栈中还有元素 弹出
if(!stack.isEmpty()) {
// 栈顶元素依次出栈并输出
int length = stack.size();
for(int i = 0; i < length; i++) {
RPNList.add(stack.pop());
}
}
5 测试运行
6 满意的代码:
在对算式的合法性进行控制时,对除法的控制:
//谢绝野蛮连成连÷
for(int e=0;e<p;e++){
if(Op[e]=='*'||Op[e]=='÷')
Op[1+e]=oper[(int) (Math.random()*2)];
if(Op[e]!='*'&&Op[e]!='÷')
Op[1+e]=oper[(int) (Math.random()*4)];
}
//sffg
//合法性问题:1 负数 2 分数.
//先对乘除处理,处理之后,归纳为简单加减运算'
/* 解决分数问题 */
for(int e=0;e<p;e++){
if(Op[e]=='÷')
Num[e]=Num[e+1]*(int)(Math.random()*20); if(Op[e]=='-')
{
Num[e+1]=(int)(Math.random()*Num[e]/8);
// Num[e+2]=(int)(Math.random()*Num[e+1]);
Num[e+2]=(int)(Math.random()*4)+1; }
}
首先避免了连÷(以及连*---这是防止数据过大),可以减少判断。然后,对整个算式进行搜索,发现符号为÷的,将除号前的被除数字乘上除数,这样永远不会出现未除尽的情况。
7 总结:完成这次作业最大的障碍就是语言问题,对于java过于陌生了些。另外,我觉得这次的作业还是可以体现模块化原则的,虽然我把所有的方法都写道一个类里面去了,但是功能是由各个组件独立完成的。
8 psp
|
|||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||||
2016011998+sw的更多相关文章
- The Nine Indispensable Rules for HW/SW Debugging 软硬件调试之9条军规
I read this book in the weekend, and decided to put the book on my nightstand. It's a short and funn ...
- 微软职位内部推荐-Sr. SW Engineer for Azure Networking
微软近期Open的职位: Senior SW Engineer The world is moving to cloud computing. Microsoft is betting Windows ...
- 重新定位svn地址的方法(windows和linux),svn switch(sw)的帮助信息
今天公司的路由器出现问题,服务器的IP地址也跟着变了,但是原来的svn已经设置好了,现在需要更换地址 查询原地址的方法: root@jack-desktop:codes# svn info 路径: . ...
- CopyU!SW新版发布!
CopyU!SW新版发布,版本号为:2.1.412.213 主要更新内容如下: 此版本(2.1.412.213)主要作了如下更新: 1.修复了CopyU!SW版本中的运行模式规则的设定错 ...
- MDK5使用Jlink下载显示 no cortex m sw divice 解决办法
问题: (1)下面界面中找不到设备 (2)下载程序的时候提示" no cortex m sw divice " 然后是"target dll has been cance ...
- 调研pwa和sw
概述 处于好奇,最近我调研了一下pwa和service worker,有些新的,记录下来,供以后开发时参考,相信对其他人也有用.pwa主要是通过service worker实现的,它主要包括桌面图标, ...
- yum解决 "Couldn't resolve host 'apt.sw.be'" 错误
1.yum无法安装工具 failure: repodata/repomd.xml from dag: [Errno 256] No more mirrors to try.http://apt. ...
- tunnel sw
tunnel sw openssh vpn httprltunnel BarbaTunnel ngrok Chisel https://github.com/jpillora/chisel/blob/ ...
- [na]二层sw数据交换
1,同vlan下,两台pc配置了GW,arp请求过程. Pc1 ping pc0的时候,触发pc1的arp请求,发给GW后,GW继续发给pc0(同一个vlan),pc0收到后给pc1回复.Pc1发出i ...
随机推荐
- PHP原生开发的各大音乐平台API接口
支持以下音乐平台 网易云音乐 QQ音乐 酷狗音乐 酷我音乐 虾米音乐 百度音乐 一听音乐 咪咕音乐 荔枝FM 蜻蜓FM 喜马拉雅FM 全民K歌 5sing原创 5sing翻唱 若是数据获取失败 方案一 ...
- Cloudera Manager Server CDH 5.15部署
安装前准备 主机和系统 Host OS Memory IP bigdata001-dev Cent OS 7.4 x64 32G 192.168.1.1 bigdata002-dev Cent OS ...
- Hadoop源码学习笔记之NameNode启动场景流程一:源码环境搭建和项目模块及NameNode结构简单介绍
最近在跟着一个大佬学习Hadoop底层源码及架构等知识点,觉得有必要记录下来这个学习过程.想到了这个废弃已久的blog账号,决定重新开始更新. 主要分以下几步来进行源码学习: 一.搭建源码阅读环境二. ...
- C++编写DLL动态链接库的步骤与实现方法
原文:http://www.jb51.net/article/90111.htm 本文实例讲述了C++编写DLL动态链接库的步骤与实现方法.分享给大家供大家参考,具体如下: 在写C++程序时,时常需要 ...
- 03 - django简介
1.MVC与MTV模型 2.Django的下载与基本命令 pip install django==2.0.1 第三方库安装到哪里了? 创建一个django project C:\Desktop\fir ...
- spring学习笔记 星球日one - xml方式配置bean
ide: idea lib包的导入:http://webcache.googleusercontent.com/search?q=cache:http://zyjustin9.iteye.com/bl ...
- 快速搭建属于自己的mongodb数据库
前言 MongoDB 是一个基于分布式文件存储的数据库.由C++语言编写,支持Windows,Linux,OSX,Solaris等平台,默认端口为27017,是一个效率非常高的nosql数据库. 我的 ...
- 使用web api开发微信公众号,调用图灵机器人接口(一)
此文将分两篇讲解,主要分为以下几步 签名校验; 首次提交验证申请; 接收消息; 被动响应消息(返回XML); 映射图灵消息及微信消息; 其实图灵机器人搭载微信公众号很简单,只需要把图灵的地址配到公众后 ...
- React——JSX
一.将表达式嵌套在JSX中 要在JSX中内嵌js表达式只需要将js表达式放在{}中,例如: const element = <h1>this is a JSX {sayName()}< ...
- JMeter下Groovy和BeanShell语言在不同组件中性能差异实践探究
一般而言JMeter下性能最好的是jar包这类java原生请求,对于JMeter并没有原生支持的请求,一般都会将其直接编译为jar包,然后再JMeter中调用,这样性能最好. 但是有些需求并不适合用j ...