结对编程 四则运算(java)(胡大华 黄绪明)

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,经过不断的学习和查找资料,最终解决问题。

    这次的结对编程,让我体会到了合作的力量,同时也认识到了自己数据结构方面知识的不足,今后会多加学习这方面的知识,完善自己的不足。

 

posted @ 2018-09-30 22:38  黄绪明  阅读(171)  评论(1编辑  收藏  举报