Github项目地址
https://github.com/yogurt1998/Myapp
项目需求
题目:
实现一个自动生成小学四则运算题目的命令行程序
功能
1.使用-n 参数控制生成题目的个数
2.使用-r 参数控制题目中数值(自然数、真分数和真分数分母)的范围
3.生成的题目中计算过程不能产生负数
4. 生成的题目中如果存在形如e1 ÷ e2的子表达式,那么其结果应是真分数。
5. 每道题目中出现的运算符个数不超过3个。
6. 程序一次运行生成的题目不能重复,即任何两道题目不能通过有限次交换+和×左右的算术表达式变换为同一道题目。
7.生成的题目存入执行程序的当前目录下的Exercises.txt文件
8.在生成题目的同时,计算出所有题目的答案,并存入执行程序的当前目录下的Answers.txt文件
9. 程序应能支持一万道题目的生成。
10. 程序支持对给定的题目文件和答案文件,判定答案中的对错并进行数量统计,输入参数如下:
Myapp.exe -e <exercisefile>.txt -a <answerfile>.txt
统计结果输出到文件Grade.txt,格式如下:
Correct: 5 (1, 3, 5, 7, 9)
Wrong: 5 (2, 4, 6, 8, 10)
PSP表格
PSP2.1 |
Personal Software Process Stages |
预估耗时(分钟) |
实际耗时(分钟) |
Planning |
计划 |
30 |
30 |
· Estimate |
· 估计这个任务需要多少时间 |
15 |
20 |
Development |
开发 |
||
· Analysis |
· 需求分析 (包括学习新技术) |
120 |
150 |
· Design Spec |
· 生成设计文档 |
30 |
40 |
· Design Review |
· 设计复审 (和同事审核设计文档) |
20 |
30 |
· Coding Standard |
· 代码规范 (为目前的开发制定合适的规范) |
20 |
30 |
· Design |
· 具体设计 |
60 |
90 |
· Coding |
· 具体编码 |
1000 |
1200 |
· Code Review |
· 代码复审 |
60 |
90 |
· Test |
· 测试(自我测试,修改代码,提交修改) |
30 |
60 |
Reporting |
报告 |
30 |
40 |
· Test Report |
· 测试报告 |
60 |
60 |
· Size Measurement |
· 计算工作量 |
20 |
20 |
· Postmortem & Process Improvement Plan |
· 事后总结, 并提出过程改进计划 |
40 |
40 |
合计 |
1535 |
1900 |
设计实现过程
设计思路
先用随机函数得到随机生成题目,为了方便计算转为后缀表达式,用堆栈的方式计算答案,用一个布尔值来判断是否可用。
利用Scanner类获取键盘输入值并判断。
用一个while循环来生成多道题目。
用字符流输入到文件中。
代码说明
1. 用随机函数randon()生产随机数以及运算符组成题目
// 生成四则运算式
public Exercise createFormula(int n) { String question1 = null; String num1, num2, num3, num4; // 四个运算数 String char1, char2, char3; // 三个运算符 char1 = randomChar();
char2 = randomChar();
char3 = randomChar(); num1 = (int)(Math.random()*2) == 0 ? Integer.toString((int)Math.round(Math.random()*(n))) : createScore(n);
num2 = (int)(Math.random()*2) == 0 ? Integer.toString((int)Math.round(Math.random()*(n))) : createScore(n);
num3 = (int)(Math.random()*2) == 0 ? Integer.toString((int)Math.round(Math.random()*(n))) : createScore(n);
num4 = (int)(Math.random()*2) == 0 ? Integer.toString((int)Math.round(Math.random()*(n))) : createScore(n); String question2 = num1 + ' ' + char1 + ' ' + num2; String question3[] = {question2 + ' ' + char2 + ' ' + num3,
'(' + question2 + ')' + ' ' + char2 + ' ' + num3,
num1 + ' ' + char1 + ' ' + '(' + num2 + ' ' + char2 + ' ' + num3 + ')'}; String question4[] = {question3[(int)(Math.random()*3)] + ' ' + char3 + ' ' + num4,
'(' + question3[(int)(Math.random()*3)] + ')' + ' ' + char3 + ' ' + num4,
num4 + ' ' + char3 + ' ' + '(' + question3[(int)(Math.random()*3)] + ')',
'(' + question2 + ')' + ' ' + char2 + ' ' + num3 + ' ' + char3 + ' ' + num4,
question2 + ' ' + char2 + ' ' + '(' + num3 + ' ' + char3 + ' ' + num4 + ')',
num1 + ' ' + char1 + ' ' + '(' + num2 + ' ' + char2 + ' ' + num3 + ')' + ' ' + char3 + ' ' + num4}; switch ((int)(Math.random()*3)) {
case 0:
question1 = question2;
break;
case 1:
question1 = question3[(int)(Math.random()*3)];
break;
case 2:
question1 = question4[(int)(Math.random()*6)];
break;
default:
break;
} Exercise exercise = new Exercise(num1, num2, num3, num4, char1, char2, char3, question1); return exercise;
} // 分数生成器
public String createScore(int n) {
int upNum, downNum;
do {
upNum = (int)Math.round(Math.random()*(n));
downNum = (int)Math.round(Math.random()*(n-1)+1);
} while (upNum / downNum > n); return upNum > downNum ? upNum / (downNum - upNum % downNum) + "'" + (upNum % downNum) + "/" + downNum
: upNum + "/" + downNum;
} // 随机运算符
private String randomChar() {
int key = (int)(Math.random()*4);
switch (key) {
case 0:
return "+";
case 1:
return "-";
case 2:
return "×";
case 3:
return "÷";
default:
break;
}
return "";
}
2. 读取和计算题目
1. 将String 类型的题目按运算数、运算符、括号转换为ArrayList<String>;
// String转为list
private ArrayList<String> stringsToList(String s) {
ArrayList<String> list = new ArrayList<>();
StringBuilder temp = new StringBuilder();
for (int i = 0; i < s.length(); ++i) {
char c = s.charAt(i);
if((c >= '0' && c <= '9') || c == '\'' || c == '/') {
temp.append(c);
} else {
String string = temp.toString();
if (!string.isEmpty()) {
list.add(string);
}
temp = new StringBuilder();
}
if (i == s.length()-1) {
String string = temp.toString();
if (!string.isEmpty()) {
list.add(string);
}
}
if (isOperator(c))
list.add(c + "");
if (c == '(' || c == ')')
list.add(c + "");
}
return list;
}
2.将中缀表达式变为后缀表达式
// 中缀表达式转为后缀表达式
private ArrayList<String> changeToPostfixEx(ArrayList<String> list) {
Stack<String> stack = new Stack<String>();
ArrayList<String> pList = new ArrayList<String>();
int level = -1;
for (int i = 0; i < list.size(); ++i) {
String aList = list.get(i);
if (!sIsOperator(aList) && !aList.equals("(") && !aList.equals( ")")) {
pList.add(aList);
} else if (aList.equals("(")) {
stack.push(aList);
level = 0;
} else if (aList.equals( ")")){
while (!stack.peek().equals("(")) {
pList.add(stack.pop());
}
stack.pop();
if (stack.empty())
level = -1;
else {
level = getLevel(stack.peek());
}
} else if (getLevel(aList) > level) {
stack.push(aList);
if (level != 0)
level = getLevel(stack.peek());
} else {
while (!stack.empty() && getLevel(aList) <= level) {
pList.add(stack.pop());
}
stack.push(aList);
level = getLevel(stack.peek());
}
}
while (!stack.empty()) {
pList.add(stack.pop());
}
return pList;
}
3. 计算后缀表达式得到答案
// 计算后缀表达式
private Score calculateHEx(ArrayList<String> list) { boolean beUse = true; // 是否使用 Stack<String> stack = new Stack<>(); for (int i = 0; i < list.size(); ++i) {
if (!sIsOperator(list.get(i))) {
stack.push(list.get(i));
} else {
Score result = new Score();
String resultS = null;
switch (list.get(i)) {
case "+":
String a1 = stack.pop();
String a2 = stack.pop();
result = addScore(a2, a1);
resultS = result.up + "/" + result.down;
stack.push(resultS);
break;
case "-":
String a3 = stack.pop();
String a4 = stack.pop();
result = subtractScore(a4, a3);
resultS = result.up + "/" + result.down;
stack.push(resultS);
break;
case "×":
String a5 = stack.pop();
String a6 = stack.pop();
result = mulScore(a6, a5);
resultS = result.up + "/" + result.down;
stack.push(resultS);
break;
case "÷":
String a7 = stack.pop();
String a8 = stack.pop();
result = divScore(a8, a7);
resultS = result.up + "/" + result.down;
stack.push(resultS);
break;
default:
break;
}
beUse = result.canBeUse;
} if (!beUse)
break;
}
Score result = new Score();
result.canBeUse = beUse;
if (result.canBeUse) {
String endResult = stack.pop();
result.changeScore(endResult);
list.add(endResult);
simplifyScore(result);
}
return result;
}
4.Score类表示所有的数,并用canBeUse属性表示是否可用。
import java.util.ArrayList; import javax.naming.InitialContext; public class Score {
int up;
int down;
boolean canBeUse = true; // ËãʽÊÇ·ñ¿ÉÓà public Score() {
// TODO Auto-generated constructor stub
} public void changeScore(String s) {
boolean isBig = false;
boolean isHave = false;
ArrayList<String> list = new ArrayList<>();
StringBuilder temp = new StringBuilder(); for (int i = 0; i < s.length(); ++i) {
char c = s.charAt(i);
if (c == '/')
isHave = true;
if (c == '\'')
isBig = true;
if (c >= '0' && c <= '9') {
temp.append(c);
} else {
list.add(temp.toString());
temp = new StringBuilder();
}
if (i == s.length() - 1) {
if (!temp.toString().isEmpty()) {
list.add(temp.toString());
}
}
} if (list.isEmpty()) {
canBeUse = false;
} else {
if (!isBig && !isHave) {
this.up = Integer.parseInt(list.get(0));
this.down = 1;
} else if (isBig) {
this.up = Integer.parseInt(list.get(0))
* Integer.parseInt(list.get(2)) + Integer.parseInt(list.get(1));
this.down = Integer.parseInt(list.get(2));
} else if (isHave && !isBig){
this.up = Integer.parseInt(list.get(0));
this.down = Integer.parseInt(list.get(1));
}
}
} }
3. IO输出
用fileWrite()和BufferWrite();
public class IO { public void writeExercise(String context, int flag) {
File exerFile = new File("D:\\test\\Exercise.txt");
File answerFile = new File("D:\\test\\Answer.txt");
if (flag == 0) {
writeIn(context, exerFile);
} else {
writeIn(context, answerFile);
}
} private void writeIn(String context, File file) {
FileWriter fileWriter;
BufferedWriter brout;
try {
fileWriter = new FileWriter(file, true);
brout = new BufferedWriter(fileWriter);
brout.write(context);
brout.flush();
brout.close(); } catch (IOException e) {
// TODO Auto-generated catch block
e.printStackTrace();
} } }
4. 主函数
import java.util.Scanner; import javax.swing.CellEditor; public class main { public static void main(String[] args) {
// TODO Auto-generated method stub String order = null;
int exNum = 0, valueNum = 1; while (valueNum <= 1) {
System.out.println("输入命令:1. 使用 -n 参数控制生成题目的个数 2. 使用 -r 参数控制题目中数值(自然数、真分数和真分数分母)的范围");
Scanner scanner = new Scanner(System.in);
if (scanner.hasNext()) {
order = scanner.next();
}
if (scanner.hasNextLine()) {
if (order.equals("-n")) {
exNum = scanner.nextInt();
} else if (order.equals("-r")) {
valueNum = scanner.nextInt();
} else {
System.out.println("请输入正确的命令参数:1. 使用 -n 参数控制生成题目的个数 2. 使用 -r 参数控制题目中数值(自然数、真分数和真分数分母)的范围");
}
} if (valueNum <= 1) {
System.err.println("请输入正确的命令参数:1. 使用 -n 参数控制生成题目的个数 2. 使用 -r 参数控制题目中数值(自然数、真分数和真分数分母)的范围");
}
}
int i = 0;
int j = 0;
IO io = new IO();
while (j < exNum) {
Func func = new Func();
Exercise exercise = func.createFormula(valueNum - 1);
Score answer = func.calculateExercise(exercise.aQuestion);
String endAnswer = null;
if (answer.canBeUse) {
if (answer.down == 1) {
endAnswer = answer.up + "";
io.writeExercise(++j + ". " + exercise.aQuestion + " = " + "\r\n", 0);
io.writeExercise(j + ". " + endAnswer + "\r\n", 1);
} else if (answer.up > answer.down) {
endAnswer = answer.up / (answer.down - answer.up % answer.down) + "'" + answer.up % answer.down + "/" + answer.down;
io.writeExercise(++j + ". " + exercise.aQuestion + " = " + "\r\n", 0);
io.writeExercise(j + ". " + endAnswer + "\r\n", 1);
}
} else {
continue;
}
} } }
测试运行
题目:
答案:
项目小结
此次的结对编程,胡大华同学负责表达式的生成、计算,程序框架的设计,我负责IO生成文件、编写博文。
刚拿到题目的时候,我们的对题目只有一点模糊的想法,经过不断地写代码实现,逐渐发现不同的问题,遇到过各种问题和BUG,经过不断的学习和查找资料,最终解决问题。
这次的结对编程,让我体会到了合作的力量,同时也认识到了自己数据结构方面知识的不足,今后会多加学习这方面的知识,完善自己的不足。