四则运算结对项目之GUI
本次结对编程让我学到了许多许多知识,受益匪浅!在此之前,我没想过我能做出一个双击运行的小程序。
感谢我的队友与我同心协力,感谢室友宇欣告诉我操作符为“最多多少”而不是“多少”并教我使用效能分析工具,感谢陈杰不辞辛苦帮我测试14寸显示屏效果,感谢福孝大佬给我发的安装包!感谢学姐对项目的建议!
代码仓库地址:.git
本人:庄莉,学号:2016012034
队友:王璐瑶,学号:2016012095
计划PSP
PSP | 任务内容 | 计划共完成需要的时间(h) |
Planning | 计划 | 0.5 |
Estimate | 估计这个任务需要多少时间,并规划大致工作步骤 | 0.5 |
Development | 开发 | 39.25 |
Analysis | 需求分析 (包括学习新技术) | 0.5 |
Design Spec | 生成设计文档 | 0.25 |
Design Review | 设计复审 (和同事审核设计文档) | 0.25 |
Coding Standard | 代码规范 (为目前的开发制定合适的规范) | 0.25 |
Design | 具体设计 | 2 |
Coding | 具体编码 | 30 |
Code Review | 代码复审 | 1 |
Test | 测试(自我测试,修改代码,提交修改) | 5 |
Reporting | 报告 | 4 |
Test Report | 测试报告(包括博客) | 3 |
Size Measurement | 计算工作量 | 0.5 |
Postmortem & Process Improvement Plan | 事后总结, 并提出过程改进计划 | 0.5 |
结对编程对接口的设计
信息隐藏(Information Hiding)
Information hiding is part of the foundation of both structured design and object-oriented design. In structured design, the notion of “black boxes” comes from information hiding. In object-oriented design, it gives rise to the concepts of encapsulation and modularity, and it is associated with the concept of abstraction.
在《代码大全》中列出了两类需要隐藏的内容:
第一类信息是复杂化的信息。对于我们的项目,我们的main函数只有一个对gui的实例化,使用者并不知道内部的运行方式。内部的算法实现封装起来,外部只有调用的接口,只可以调用方法,不可以改变内部变量,做到了信息隐藏。
对于第二类,是指变动的信息。比如在用户的输入需求中出现了错误,提示并返回,这个错误在类中进行了适当的处理,错误没有扩散,这样可以提高程序的容错性。
接口设计(Interface Design)
在本项目设计接口过程中,按需求新建接口,使用明确的命名方式使接口的功能清晰化,增强了可读性;接口与接口之间互相独立,使用方便。
松耦合(Loose coupling)
耦合的强度依赖于:(1)一个模块对另一个模块的调用;(2)一个模块向另一个模块传递的数据量;(3)一个模块施加到另一个模块的控制的多少;(4)模块之间接口的复杂程度。等等。
模块内子程序(下一个层次上)应共享数据(有一定的耦合度),而减少全局变量能降低子程序性间的耦合性。
类与类之间通常通过接口的契约实现服务提供者/服务请求者模式,这就是典型的松耦合。
耦合程度越高,模块与模块之间的联系性就更高,系统的灵活性就越低,报错率就更高。在我们的项目中,计算模块的调用都比较单一,没有双向调用,使用之间互不干扰,增加了灵活性。
计算模块接口的设计与实现过程
经过商讨,我们决定基于我的个人项目修改。我先删除了原来的分数运算,在将普通四则运算与括号四则运算拆分,变成简单加减、四则运算、有括号加减与有括号四则运算。如图我分为5个类(test为单元测试)。
- Command类:命令行测试类,负责接收命令行的参数并启动程序。
- fileCreate类:创建文件类,负责产生result.text文件,将练习题写入文件以及做题模式的生成记录。
- formula类:式子类,负责根据调用产生同种类型的式子,含有AddSubtract(加减运算)、arithmetic(简单四则运算)、Bracket(带括号的四则运算)、Bracket_AS (带括号的加减运算)四种函数。
- calculate类:计算类,负责各种计算,含有有条件产生后一位数、有条件操作符等7个方法。
- stack类:栈,负责计算式子,并判断式子合法性。
其中,有条件生成操作符与后一位数我较为满意,它大大的降低了运行效率,部分代码可看第5模块的性能改进模块。
计算模块接口部分的性能改进
基于原来的个人项目代码,由于出现了运算过程以及运算结果数值范围的限制,原本的result(String temp)不再使用,改用了栈运算。
1 // 计算结果 2 public static Object result(String temp) { 3 ScriptEngineManager sem = new ScriptEngineManager(); 4 ScriptEngine se = sem.getEngineByName("js"); 5 Object last = 0; 6 try { 7 last = se.eval(temp); 8 } catch (ScriptException e) { 9 e.printStackTrace(); 10 } 11 return last; 12 }result函数
在栈的运算中加入判断
1 if (Math.abs(sresulat) > upper || Math.abs(sresulat) < lower) 2 { 3 return 0; 4 }判断
而对于简单加减无括号全程不改变优先级的运算则不过栈,直接边生成数字便运算,减少了运算时间。
另外,原本的操作符是一开始随机生成好的再判断选择后一个数,然后再判断符号是否合法,再修改符号,如果还是有小数或负数,则重新运行生成算式的函数,这样使得代码运行有些慢且多次运行。再加上数值范围的限定以及可以存在负数,我改变了想法。
因为负数的存在,使得加减号并没有数字的限制,而乘法有上限限制,除法有下限限制。所以在只有加减的运算中,符号随机生成,后一个数根据运算符以及数值范围生成合法的数。
1 // 相加不超过范围 2 public static int decide0(int x, int min, int max) 3 { 4 int y; 5 int temp = 0; 6 if (x > 0) 7 { 8 temp = max - min - x + 1;// 加一个正整数范围 9 } else 10 { 11 temp = max - (min - x) + 1;// 加至正整数的范围 12 } 13 if (temp < 0) 14 {// 范围小于0 15 if (x > 0) 16 { 17 temp = Math.abs(x) - min * 2 + 1;// 正整数过大,需加负数 18 y = 0 - (int) (Math.random() * temp) - min; 19 } else 20 { 21 temp = Math.abs(x) - 2 * min + 1;// 负数过小,越值,加小整数至负数范围 22 y = (int) (Math.random() * temp) + min; 23 } 24 } else 25 { 26 y = (int) (Math.random() * temp + min); 27 } 28 return y; 29 } 30 31 // 相减不小于最小 32 public static int decide1(int x, int min, int max) 33 { 34 int temp = 0; 35 int y = 0; 36 if (x > 0) 37 { 38 temp = x - 2 * (min - 1) - 1; // 减一个正数范围 39 } else 40 { 41 temp = max + x - min + 1;// 减一个正数范围 42 } 43 if (temp > 0) 44 { 45 if (x < 0 && temp < min) 46 { 47 temp = Math.abs(x) - 2 * min + 1;// 负数过小,需减负数 48 y = 0 - (int) (Math.random() * temp) - min; 49 } else 50 { 51 y = (int) (Math.random() * temp + min); 52 } 53 } else 54 { 55 temp = max - x - min + 1;// 只有x>0的情况会出现,正数过小,需减负数 56 y = 0 - (int) (Math.random() * temp) - min; 57 } 58 return y; 59 }加减法的后一位数选定
当有乘除时,则根据上一个数生成操作符,再根据操作符生成合法的后一位数。
1 // 操作符的选定 2 public static int operator(int num, int middle2, int middle3) 3 { 4 if (Math.abs(num) <= middle2) 5 {// 除法下界 6 if (Math.abs(num) < middle3) 7 { 8 return 3; 9 } else 10 { 11 return 0; 12 } 13 } else if (Math.abs(num) >= middle3) 14 {// 乘法上界 15 return 2; 16 } else 17 { 18 return (int) (Math.random() * 4); 19 } 20 } 21 // 下一位数字的选定 22 public static int[] numberB(int key, int num, int lower, int upper) 23 { 24 int[] find = new int[] { 0, lower }; 25 if (key == 0) 26 { 27 find[1] = decide0(num, lower, upper); 28 return find; 29 } else if (key == 2) 30 { 31 int[] judge = new int[2]; 32 judge = decide2(num, lower);// 确保能够整除,并不低于下限 33 if (judge[0] == 0) 34 { 35 find[1] = judge[1]; 36 return find; 37 } else 38 { 39 find[0] = 1; 40 } 41 } else if (key == 3) 42 { 43 find[1] = decide3(num, lower, upper); 44 if (find[0] == 0) 45 { 46 return find; // 乘法不超过上限 47 } 48 } 49 find[1] = decide1(num, lower, upper); 50 return find; 51 }操作符选定以及下一位数字的选定
这样大大减少了重新调用函数的问题,并且实现了运算过程与数值皆在范围内的功能。
在附加题记录用户模块,一开始使用contains(name)函数判断用户,后来发现这样会出现abc与abcabc被认为同一个人而的情况,经过思考,我们使用字符串的断开。
1 String[] arrays = txt.split(" ");
再使用equals(String)函数判断用户,解决了这个问题。
其中,生成有括号与乘除的式子生成的函数判断耗时最多,因为它的判断较多,限制较多,优先级易改变,容易生成最终不合法的式子而重新运行。
1 // 带括号的四则运算 2 public static String Bracket(int lower, int upper, int o) { 3 int middle2 = lower * lower;// 除法下界 4 int middle3 = upper / lower;// 乘法上界 5 int brack_left = 0; // 记录未匹配的左括号个数 6 int brack = 0; // 括号个数 7 int j = 0; 8 char[] p = new char[] { '+', '-', '÷', '*' }; 9 String temp1 = ""; 10 int[] num = new int[o + 1]; // 数字 11 int[] key = new int[o]; // 符号所在的下标 12 num[0] = (int) (Math.random() * (upper - lower + 1) + lower); 13 int result; 14 int[] find = new int[2]; 15 for (j = 0; j < (o - 1); j++) { 16 if (num[j] < 0) { 17 temp1 += "(" + String.valueOf(num[j]) + ")"; 18 } else { 19 temp1 += String.valueOf(num[j]); 20 } 21 int tmpcnt = brack_left; 22 for (int i = 0; i < tmpcnt; i++) { // 若当前有未匹配的左括号,则对每一个未匹配的左括号,都有一定概率生成相应右括号。 23 if ((int) (Math.random() * 5) > 1) { // 生成右括号概率为0.6 24 brack_left--; 25 temp1 += ")"; 26 } 27 } 28 key[j] = calculate.operator(num[j], middle2, middle3); 29 find = calculate.numberB(key[j], num[j], lower, upper); 30 if (find[0] == 1) { 31 key[j] = 1; 32 } 33 num[j + 1] = find[1]; 34 temp1 += String.valueOf(p[key[j]]); 35 if (((brack * 2) <= o) && (((int) (Math.random() * 2)) == 0)) { // 以一定概率生成左括号,概率为1/2 36 temp1 += "("; 37 brack++; 38 brack_left++; 39 j++; 40 if (num[j] < 0) { 41 temp1 += "(" + String.valueOf(num[j]) + ")"; 42 } else { 43 temp1 += String.valueOf(num[j]); 44 } // 生成左括号后必须生成一个数字和运算符,不然可能出现(15)这样的错误 45 key[j] = calculate.operator(num[j], middle2, middle3); 46 find = calculate.numberB(key[j], num[j], lower, upper); 47 if (find[0] == 1) { 48 key[j] = 1; 49 } 50 num[j + 1] = find[1]; 51 temp1 += p[key[j]]; 52 } 53 } 54 while (j != o) { // 判断是否为最后一个数 55 if (num[j] < 0) { 56 temp1 += "(" + String.valueOf(num[j]) + ")"; 57 } else { 58 temp1 += String.valueOf(num[j]); 59 } 60 key[j] = calculate.operator(num[j], middle2, middle3); 61 temp1 += p[key[j]]; 62 find = calculate.numberB(key[j], num[j], lower, upper); 63 if (find[0] == 1) { 64 key[j] = 1; 65 } 66 j++; 67 num[j] = find[1]; 68 } 69 if (num[o] < 0) { 70 temp1 += "(" + String.valueOf(num[o]) + ")"; 71 } else { 72 temp1 += String.valueOf(num[o]); 73 } 74 while ((brack_left) != 0) { // 补全右括号 75 temp1 += ")"; 76 brack_left--; 77 } 78 result = stack.work(temp1, lower, upper, 1); 79 if (result == 0) { 80 temp1 = Bracket(lower, upper, o); 81 } 82 return temp1; 83 84 } 85 86 }有括号四则运算
项目总体分析图,从内存,多线程,CPU等方面分析了计算模块的性能,截图如下:
性能分析过程截图:
按F4,出现以下截图。资源全部被回收。证明没有资源泄露。程序性能良好。
使用单元测试的CPU分析如下图:
使用Command.java的CPU效能分析如下图:
单元测试
1 @Test 2 public void testWork() { 3 assertEquals(0, stack.work("7-5÷(1*37)÷(1*83)", 1, 900, 1)); 4 assertEquals(30, stack.work("55+(-25)÷5*(20-15)", 2, 300, 1)); 5 assertEquals(80, stack.work("((55+25)÷5)*(20-15)", 2, 300, 1)); 6 assertEquals(0, stack.work("60*(20-15)", 2, 200, 1)); 7 }栈的测试
第一个断言测试的是无法整除返回错误标志0;
第二个断言测试的是负数运算;
第三个断言测试的是特殊括号位置的运算;
第四个断言测试的是超过数值返回错误标志0。
1 @Test 2 public void testAll() { 3 // 顺序不同以及异常测试。生成的文件会被覆盖。 4 String[] arg0 = new String[] { "-n", "100", "-m", "5", "100", "-o", "3", "-c", "-b" }; 5 String[] arg1 = new String[] { "-m", "5", "50", "-o", "3", "-n", "100", "-c" }; 6 String[] arg2 = new String[] { "-o", "3", "-m", "5", "50", "-n", "100", "-b" }; 7 String[] arg3 = new String[] { "-n", "100", "-o", "3", "-m", "5", "50" }; 8 Command.main(arg0);// 有括号四则运算测试 9 Command.main(arg1);// 四则运算测试 10 Command.main(arg2);// 有括号加减运算测试 11 Command.main(arg3);// 加减运算测试 12 }命令行正确输入测试
该部分测试的命令行的更改输入顺序的四种出题选择正常运行。输入异常部分请看第七点。
命令行单元测试覆盖率截图如下:
1 @Test 2 public void testDecide2() { 3 int[] find = new int[2]; 4 find = calculate.decide2(20, 2); 5 assertEquals(2, find[1]); 6 find = calculate.decide2(13, 2); 7 assertEquals(1, find[0]); 8 }除法选择除数测试
decide2(int x, int min)为除法选择除数的函数,函数如下:
1 // 被除数能被除数整除并不低于最小 2 public static int[] decide2(int x, int min) 3 { 4 int[] judge = new int[] { 1, 0 }; 5 int temp = Math.abs(x) / min - min + 1;// 除数的范围 6 for (int i = min; i < (temp + min); i++) 7 { 8 if (Math.abs(x) % i == 0) 9 {// 判断是否整除 10 judge[0] = 0; 11 judge[1] = i; 12 return judge; 13 } 14 } 15 return judge; 16 }decide2函数
其中,judge[0]用于判断该数能否有可整除的除数,1为没有,0为有,judge[1]为除数的值。该单元测试则测试了一次可产生除数与一次不能产生除数的情况。
异常说明
1 @Test 2 public void testAll() { 3 String[] arg4 = new String[] { "-o", "3", "-m", "5", "50", "-n" }; 4 String[] arg4_1 = new String[] { "-o", "3", "-n", "-m", "5", "50" }; 5 String[] arg4_2 = new String[] { "-n", "100000", "-m", "5", "50" }; 6 String[] arg4_3 = new String[] { "-o", "3", "-m", "5", "50" }; 7 8 String[] arg5 = new String[] { "-n", "50" }; 9 String[] arg5_1 = new String[] { "-m", "5", "-n", "50", "-o", "3" }; 10 String[] arg5_2 = new String[] { "-n", "50", "-m", "3" }; 11 String[] arg5_3 = new String[] { "-n", "50", "-o", "3", "-m" }; 12 String[] arg5_4 = new String[] { "-m", "-n", "50" }; 13 14 String[] arg6 = new String[] { "-o", "11", "-m", "5", "50", "-n", "100" }; 15 String[] arg6_1 = new String[] { "-n", "100", "-o", "-m", "5", "50" }; 16 String[] arg6_2 = new String[] { "-n", "100", "-m", "5", "50", "-o" }; 17 18 String[] arg7 = new String[] { "-m", "5", "20", "-n", "100", "-c" }; 19 String[] arg7_1 = new String[] { "-m", "5", "50", "-n", "100", "-b" }; 20 21 String[] arg8 = new String[] { "-b", "1", "-o", "3", "-m", "5", "50", "-n", "100" }; 22 String[] arg8_1 = new String[] { "-c", "1", "-o", "3", "-m", "5", "50", "-n", "100" }; 23 String[] arg8_2 = new String[] { "-n", "100", "-m", "5", "50", "-d" }; 24 25 Command.main(arg4);// 缺少题数值测试 26 Command.main(arg4_1); 27 Command.main(arg4_2);// 题数值过大测试 28 Command.main(arg4_3);// 缺少题数测试 29 30 Command.main(arg5);// 缺少数值范围 31 Command.main(arg5_1);// 缺少数值范围上限测试 32 Command.main(arg5_2); 33 Command.main(arg5_3);// 缺少数值范围上下限测试 34 Command.main(arg5_4); 35 36 Command.main(arg6);// 操作符数值过大测试 37 Command.main(arg6_1);// 缺少操作符数值测试 38 Command.main(arg6_2); 39 40 Command.main(arg7);// 乘除需要上界大于下界的平方 41 Command.main(arg7_1);// 括号需要操作符数大于1 42 43 Command.main(arg8);// 输入非法测试之b后有数字 44 Command.main(arg8_1);// 输入非法测试之c后有数字 45 Command.main(arg8_2);// 输入非法测试之无辨识字符 46 }命令行异常输入测试
对于命令行可能出现的异常大概有13个:
- 缺少题数值(-n后无带数字,如arg4与arg4_1)时,提醒缺少题数值,并告知-n的范围;
- 题数值过大(-n后数值超过10000,如arg4_2)时,提醒告知题数值范围(过小同理);
- 缺少题数(命令中无-n,如arg4_3)时,提醒-n为必须项,并告知-n范围。
- 缺少数值范围(命令中无-m,如arg5)时,提醒-m为必须项,并告知-m上下限各自范围;
- 缺少数值范围上限(-m后只带一个数字,如arg5_1和 arg5_2)时,提醒缺少上限,并告知上限范围;
- 缺少数值范围上下限(-m后不带数字,如arg5_3和 arg5_4)时,提醒缺少上下限,并告知上下限各自范围;
- 数值范围数值过小过大时,提醒告知操作符数值范围。
- 操作符数值过大(-o后数值超过10,如arg6)时,提醒告知操作符数值范围(过小同理);
- 缺少操作符数值(输入-o,后方没有带数值,如arg6_1与arg6_2)时,提醒缺少操作符数值,并告知-o范围。
- 选择乘除法但是上界小于下界的平方,无法生成含有乘除的式子(如arg7)时,提醒上界需大于下界的平方;
- 选择括号但是操作符默认为1或选择为1,不符合生成括号的条件(如arg7_1)时,提醒选择括号需要操作符数大于1。
- –b(或-c)后带数字(如arg8与arg8_1),提醒-b(或-c)后不能带数字;
- 出现除m、n、o、b、c外的字符如d等(如arg8_2),提醒输入值非法。
界面模块的详细设计过程
设计图如下:
我们先从选择出题或做题开始。
选择出题则进入出题参数输入界面。
利用MouseListener的mouseEntered(MouseEvent e)与setTitle(String);使得鼠标移到参数上,标题会有提示功能。
输入完毕点击确认后,由输入的参数判断是否有异常并提示直至无异常创建文件。
1 public class submitListener implements ActionListener { 2 public void actionPerformed(ActionEvent e) { 3 String m = "题数与数值上下限为必填项,请按标题提示输入正整数!"; 4 String m2 = "创建文件成功!"; 5 int n0, lower0, upper0, o0, c0, b0; 6 o0 = 1; 7 c0 = 0; 8 b0 = 0; 9 String o1 = ""; 10 try { 11 n0 = Integer.parseInt(n.getText()); 12 lower0 = Integer.parseInt(lower.getText()); 13 upper0 = Integer.parseInt(upper.getText()); 14 if (n0 < 1 || n0 > 10000) { 15 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), "题数范围为1-10000", "提示", 16 JOptionPane.INFORMATION_MESSAGE); 17 return; 18 } 19 if (lower0 < 1 || lower0 > 100) { 20 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), "下界范围为1-100", "提示", 21 JOptionPane.INFORMATION_MESSAGE); 22 return; 23 } 24 if (upper0 < 50 || upper0 > 1000) { 25 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), "上界范围为50-1000", "提示", 26 JOptionPane.INFORMATION_MESSAGE); 27 return; 28 } 29 if (upper0 < (2 * lower0)) { 30 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), "上界必须大于两倍下界", "提示", 31 JOptionPane.INFORMATION_MESSAGE); 32 return; 33 } 34 } catch (NumberFormatException e2) { 35 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), m, "提示", JOptionPane.INFORMATION_MESSAGE); 36 return; 37 } 38 try { 39 o1 = o.getText(); 40 o0 = Integer.parseInt(o1); 41 } catch (NumberFormatException e2) { 42 if (!o1.equals("")) { 43 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), "请输入1-10的正整数或不输入保持默认,默认为1", "提示", 44 JOptionPane.INFORMATION_MESSAGE); 45 return; 46 } 47 } 48 if (c.isSelected()) { 49 c0 = 1; 50 } 51 if (b.isSelected()) { 52 b0 = 1; 53 } 54 if (o0 == 1 && b0 == 1) { 55 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), "括号需要操作符数量大于1", "提示", 56 JOptionPane.INFORMATION_MESSAGE); 57 return; 58 } 59 if (c0 == 1 && upper0 < (lower0 * lower0)) { 60 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), "乘除法需要上界数值大于下界的平方", "提示", 61 JOptionPane.INFORMATION_MESSAGE); 62 return; 63 } 64 createFile.fileCreate(n0, lower0, upper0, o0, c0, b0); 65 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), m2, "提示", JOptionPane.INFORMATION_MESSAGE); 66 System.exit(0); 67 } 68 }参数确认
选择做题则先输入做题人名字(在这里建议使用英文,中文名字无法很好的记录)。
接着上传文件,在这里使用了txt文件过滤器,使之仅可上传txt文件。
1 FileFilter filter = new FileNameExtensionFilter("Text file", "txt"); 2 JFileChooser fileChooser = new JFileChooser(); 3 fileChooser.setAcceptAllFileFilterUsed(false); 4 fileChooser.addChoosableFileFilter(filter); 5 FileSystemView fsv = FileSystemView.getFileSystemView();过滤器
另外,出题与做题都统一为utf-8编码,免去执行文件编码错误。
BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(file, true), "UTF-8"));
InputStreamReader read = new InputStreamReader(new FileInputStream(file), "utf-8");
上传成功后开始计时做题,并于最后结果中显示用时。
经过后面JTextPane控件的启发,我考虑到出题时题目长度有长短,为了更加美观显示,应该需要自动换行,我同样采用了HTML编辑文本的想法,做出改进。
1 public static void JlabelSetText(JLabel jLabel, String longString) throws InterruptedException { 2 StringBuilder builder = new StringBuilder("<html>"); 3 char[] chars = longString.toCharArray(); 4 FontMetrics fontMetrics = jLabel.getFontMetrics(jLabel.getFont()); 5 int start = 0; 6 int len = 0; 7 while (start + len < longString.length()) { 8 while (true) { 9 len++; 10 if (start + len > longString.length()) 11 break; 12 if (fontMetrics.charsWidth(chars, start, len) > jLabel.getWidth()) { 13 break; 14 } 15 } 16 builder.append(chars, start, len - 1).append("<br/>"); 17 start = start + len - 1; 18 len = 0; 19 } 20 builder.append(chars, start, longString.length() - start); 21 builder.append("</html>"); 22 jLabel.setText(builder.toString()); 23 }换行
计时方面我原本采用秒数计时,后来考虑到当做题时间较长时,秒数很难清晰明确的表达,所以改用了 hh:mm:ss 法显示。
1 public static String getTimeStrBySecond(long second) { 2 if (second <= 0) { 3 return "00:00:00"; 4 } 5 int hours = (int) second / HOUR_SECOND; 6 if (hours > 0) { 7 8 second -= hours * HOUR_SECOND; 9 } 10 int minutes = (int) second / MINUTE_SECOND; 11 if (minutes > 0) { 12 second -= minutes * MINUTE_SECOND; 13 } 14 return (hours > 10 ? (hours + "") 15 : ("0" + hours) + ":" + (minutes > 10 ? (minutes + "") : ("0" + minutes)) + ":" 16 + (second > 10 ? (second + "") : ("0" + second))); 17 }秒数转换
自动换行处理与秒数转换被我写入新类——dataDeal类中。
最终做完题目后除了显示用时,还显示题数、分数、错题以及该题正确答案,非首次用户会显示历史分数以及最高分数。
原本该部分使用了JTextArea控件,但学姐建议正确答案部分对齐显示会更加美观,并提出了C#中的ListView控件,但很遗憾,Java中似乎并没有。JTextArea控件是纯文本显示,很难做到不同的对齐方式,所以我删除了该类。经过多方学习比较,我最终选择了JTextPane控件,该控件简单易用,可将文本显示为HTML文本,大大提高了编辑的样式性。我最终采取了表格法对齐,另外对重点突出的地方加粗变红显示,达到强调与一定视觉冲击效果,可从后文看到对比图。
1 String text = "<p style='font-family:楷体; font-size:19'>" + name + " 本次用时<span style='color:red'><strong> " 2 + dataDeal.getTimeStrBySecond(spentTime) + " </strong></span>,得分<span style='color:red'><strong> " 3 + goal + " </strong></span>分。<br>"; 4 if (size0 == 0) { 5 text += "你总共答了<span style='color:red'><strong> " + size 6 + " </strong></span>道题,并全部答对!<span style='color:red'><strong>恭喜!</strong></span></p>"; 7 } else { 8 text += "你总共答了<span style='color:red'><strong> " + size 9 + " </strong></span>道题,答对<span style='color:red'><strong> " + size1 10 + " </strong></span>道,答错<span style='color:red'><strong> " + size0 11 + " </strong></span>道,分别为:</p><p><table border=0>"; 12 for (int i = 0; i < (size0 * 2); i++) { 13 text += "<tr><td style='font-family:楷体; font-size:19'><strong>" + wrong.get(i++) 14 + " </strong></td><td width='180' style='font-family:楷体; font-size:19;color:red'><strong> " 15 + wrong.get(i) + "</strong></td></tr>"; 16 } 17 } 18 text += "</table></p>"; 19 text += "<p style='font-family:楷体; font-size:19'>" + createFile.record(name, goal) + "</p>"; 20 21 JTextPane textarea = new JTextPane(); 22 textarea.setContentType("text/html"); 23 textarea.setText(text); 24 textarea.setEditable(false); 25 JScrollPane textAreascrollPane = new JScrollPane(textarea); 26 add(textAreascrollPane, BorderLayout.CENTER);JTextPane
界面模块与计算模块的对接
如图所示
在界面模块选择出题输入参数之后调用fileCreate类,再由fileCreate类调用计算模块,创建result.txt
在界面模块选择做题输入名字、上传文件、做题。做题时调用计算模块的stack类计算判断正确性,记录错题。最终结果由计算模块中的fileCreate类的record(String name, int goal)记录,由界面模块显示。
实现的功能大致有12个,并且为了提高用户体验,修改了图标并增加了背景,将操作符数修改为下拉框选择,默认选择为1,避免输入非数字错误:
模式选择
出题参数输入(前后对比图)
出题参数要求提醒
输入参数有误提醒(见第七点异常)
生成文件
记录用户
上传文件(只允许txt文件)
判断文件是否为空或非练习题
计时
一道一道做题并且题目过长时自动换行
评分
根据学姐给的建议做出了修改,以下为前后对比图,正确答案对齐,使之更加美观。另外我修改了做题时间的显示形式,这样当做题时间较长时可以更加清晰的看出时间情况。而做题时间、得分情况、错题与正确答案皆加粗甚至标红,使之更加显眼,提高用户体验。
记录历史分数与最高分数
结对编程
我们先一起分析了需求与功能的实现,并提出了一些有实质性的方法,并确认数据的传递方式。再分析各自的个人项目代码,指出了双方优劣性,在综合考虑选择基础代码加以改进。
我们根据自己较为擅长的方面分工,如相对之下,我对gui较为熟悉,而她对字符串处理较为熟悉,则我负责界面展示而她负责命令行的分析。各自写完之后我们再复审双方代码,对代码不理解之处询问并补充注释,以及对双方异常情况补充。最后在一起整合双方代码,使之成为完整项目。
结对编程的优缺点
在此过程中我们互相帮助、互相学习、能力上得到互补,而代码和产品质量提高,有效地减少bug并且考虑到更多方面的情况。有两台电脑可以测试程序效果,如她的电脑比我小,我的gui显示不同,她的部分算式被遮挡,最终我选择了将按钮部分的面板设为透明,解决了这个问题。
不足之处在于队友之间的进度相互影响,不同的代码风格之间的磨合也花费了一定时间。
双方优缺点:
庄莉 | 王璐瑶 | |
优点 | 认真细心,有责任心 | 任劳任怨 |
代码能力高 | 对字符串以及字符串数组的处理十分熟练 | |
动手能力强 | 很有想法,有好点子 | |
缺点 | 有时候对于小问题过于钻牛角尖 | 因生病而不在状态,没注意到比较细的地方,时间较少 |
实际PSP
PSP | 任务内容 | 实际完成需要的时间(min) |
Planning | 计划 | 0.5 |
Estimate | 估计这个任务需要多少时间,并规划大致工作步骤 | 0.5 |
Development | 开发 | 53.25 |
Analysis | 需求分析 (包括学习新技术) | 0.5 |
Design Spec | 生成设计文档 | 0.25 |
Design Review | 设计复审 (和同事审核设计文档) | 0.25 |
Coding Standard | 代码规范 (为目前的开发制定合适的规范) | 0.25 |
Design | 具体设计 | 1 |
Coding | 具体编码 | 40 |
Code Review | 代码复审 | 1 |
Test | 测试(自我测试,修改代码,提交修改) | 10 |
Reporting | 报告 | 9 |
Test Report | 测试报告 | 8 |
Size Measurement | 计算工作量 | 0.5 |
Postmortem & Process Improvement Plan | 事后总结, 并提出过程改进计划 | 0.5 |
本次结对编程真的让我学到很多知识,尤其是各种操作,就像上一篇博客的链接一样,我查了许许多多这样的链接,学习了一种有一种的方法,与队友配合,完成了这次项目。而每次写博客,都能重新总结我的思路,受益良多。
虽然真的很辛苦,但能做出来也就够了。
以下部分由于时间与精力关系,我们小组并没有完成,仅提供思路参考,有想法的同学可加以尝试。
附加题多语言思路参考:程序国际化
思路注意点参考:
- 文字描述使用短语、名词与阿拉伯数字,便于翻译。 标准语言建议英语,中文字符难以使用字符串判断内容。
- 语言选择可为下拉框列表旁边带一不可编辑的文本框。由于可以自添加,所以可以建立一个文件(如txt),文件中加入选项,执行代码时读出文件,for循环将每一项添加进下拉框列表中。有一项设置为其他或自定义,选择它时,旁边的文本框变为可编辑,用于添加语言(也可让下拉框为可编辑,当输入的语言下拉选项中没有时认为添加新语言)。
- 添加语言则需配置文件(参考网址),将所有需要翻译的文字让用户对应翻译。生成新的配置文件,并命名为该语言(使用英文命名),将该语言添加到保存下拉框选项的文件中。
- 当用户选择一种语言时,通过其相同的命名,即可调用该语言的配置文件进行翻译,达到多语言转化功能。
转载于:.html
四则运算结对项目之GUI
本次结对编程让我学到了许多许多知识,受益匪浅!在此之前,我没想过我能做出一个双击运行的小程序。
感谢我的队友与我同心协力,感谢室友宇欣告诉我操作符为“最多多少”而不是“多少”并教我使用效能分析工具,感谢陈杰不辞辛苦帮我测试14寸显示屏效果,感谢福孝大佬给我发的安装包!感谢学姐对项目的建议!
代码仓库地址:.git
本人:庄莉,学号:2016012034
队友:王璐瑶,学号:2016012095
计划PSP
PSP | 任务内容 | 计划共完成需要的时间(h) |
Planning | 计划 | 0.5 |
Estimate | 估计这个任务需要多少时间,并规划大致工作步骤 | 0.5 |
Development | 开发 | 39.25 |
Analysis | 需求分析 (包括学习新技术) | 0.5 |
Design Spec | 生成设计文档 | 0.25 |
Design Review | 设计复审 (和同事审核设计文档) | 0.25 |
Coding Standard | 代码规范 (为目前的开发制定合适的规范) | 0.25 |
Design | 具体设计 | 2 |
Coding | 具体编码 | 30 |
Code Review | 代码复审 | 1 |
Test | 测试(自我测试,修改代码,提交修改) | 5 |
Reporting | 报告 | 4 |
Test Report | 测试报告(包括博客) | 3 |
Size Measurement | 计算工作量 | 0.5 |
Postmortem & Process Improvement Plan | 事后总结, 并提出过程改进计划 | 0.5 |
结对编程对接口的设计
信息隐藏(Information Hiding)
Information hiding is part of the foundation of both structured design and object-oriented design. In structured design, the notion of “black boxes” comes from information hiding. In object-oriented design, it gives rise to the concepts of encapsulation and modularity, and it is associated with the concept of abstraction.
在《代码大全》中列出了两类需要隐藏的内容:
第一类信息是复杂化的信息。对于我们的项目,我们的main函数只有一个对gui的实例化,使用者并不知道内部的运行方式。内部的算法实现封装起来,外部只有调用的接口,只可以调用方法,不可以改变内部变量,做到了信息隐藏。
对于第二类,是指变动的信息。比如在用户的输入需求中出现了错误,提示并返回,这个错误在类中进行了适当的处理,错误没有扩散,这样可以提高程序的容错性。
接口设计(Interface Design)
在本项目设计接口过程中,按需求新建接口,使用明确的命名方式使接口的功能清晰化,增强了可读性;接口与接口之间互相独立,使用方便。
松耦合(Loose coupling)
耦合的强度依赖于:(1)一个模块对另一个模块的调用;(2)一个模块向另一个模块传递的数据量;(3)一个模块施加到另一个模块的控制的多少;(4)模块之间接口的复杂程度。等等。
模块内子程序(下一个层次上)应共享数据(有一定的耦合度),而减少全局变量能降低子程序性间的耦合性。
类与类之间通常通过接口的契约实现服务提供者/服务请求者模式,这就是典型的松耦合。
耦合程度越高,模块与模块之间的联系性就更高,系统的灵活性就越低,报错率就更高。在我们的项目中,计算模块的调用都比较单一,没有双向调用,使用之间互不干扰,增加了灵活性。
计算模块接口的设计与实现过程
经过商讨,我们决定基于我的个人项目修改。我先删除了原来的分数运算,在将普通四则运算与括号四则运算拆分,变成简单加减、四则运算、有括号加减与有括号四则运算。如图我分为5个类(test为单元测试)。
- Command类:命令行测试类,负责接收命令行的参数并启动程序。
- fileCreate类:创建文件类,负责产生result.text文件,将练习题写入文件以及做题模式的生成记录。
- formula类:式子类,负责根据调用产生同种类型的式子,含有AddSubtract(加减运算)、arithmetic(简单四则运算)、Bracket(带括号的四则运算)、Bracket_AS (带括号的加减运算)四种函数。
- calculate类:计算类,负责各种计算,含有有条件产生后一位数、有条件操作符等7个方法。
- stack类:栈,负责计算式子,并判断式子合法性。
其中,有条件生成操作符与后一位数我较为满意,它大大的降低了运行效率,部分代码可看第5模块的性能改进模块。
计算模块接口部分的性能改进
基于原来的个人项目代码,由于出现了运算过程以及运算结果数值范围的限制,原本的result(String temp)不再使用,改用了栈运算。
1 // 计算结果 2 public static Object result(String temp) { 3 ScriptEngineManager sem = new ScriptEngineManager(); 4 ScriptEngine se = sem.getEngineByName("js"); 5 Object last = 0; 6 try { 7 last = se.eval(temp); 8 } catch (ScriptException e) { 9 e.printStackTrace(); 10 } 11 return last; 12 }result函数
在栈的运算中加入判断
1 if (Math.abs(sresulat) > upper || Math.abs(sresulat) < lower) 2 { 3 return 0; 4 }判断
而对于简单加减无括号全程不改变优先级的运算则不过栈,直接边生成数字便运算,减少了运算时间。
另外,原本的操作符是一开始随机生成好的再判断选择后一个数,然后再判断符号是否合法,再修改符号,如果还是有小数或负数,则重新运行生成算式的函数,这样使得代码运行有些慢且多次运行。再加上数值范围的限定以及可以存在负数,我改变了想法。
因为负数的存在,使得加减号并没有数字的限制,而乘法有上限限制,除法有下限限制。所以在只有加减的运算中,符号随机生成,后一个数根据运算符以及数值范围生成合法的数。
1 // 相加不超过范围 2 public static int decide0(int x, int min, int max) 3 { 4 int y; 5 int temp = 0; 6 if (x > 0) 7 { 8 temp = max - min - x + 1;// 加一个正整数范围 9 } else 10 { 11 temp = max - (min - x) + 1;// 加至正整数的范围 12 } 13 if (temp < 0) 14 {// 范围小于0 15 if (x > 0) 16 { 17 temp = Math.abs(x) - min * 2 + 1;// 正整数过大,需加负数 18 y = 0 - (int) (Math.random() * temp) - min; 19 } else 20 { 21 temp = Math.abs(x) - 2 * min + 1;// 负数过小,越值,加小整数至负数范围 22 y = (int) (Math.random() * temp) + min; 23 } 24 } else 25 { 26 y = (int) (Math.random() * temp + min); 27 } 28 return y; 29 } 30 31 // 相减不小于最小 32 public static int decide1(int x, int min, int max) 33 { 34 int temp = 0; 35 int y = 0; 36 if (x > 0) 37 { 38 temp = x - 2 * (min - 1) - 1; // 减一个正数范围 39 } else 40 { 41 temp = max + x - min + 1;// 减一个正数范围 42 } 43 if (temp > 0) 44 { 45 if (x < 0 && temp < min) 46 { 47 temp = Math.abs(x) - 2 * min + 1;// 负数过小,需减负数 48 y = 0 - (int) (Math.random() * temp) - min; 49 } else 50 { 51 y = (int) (Math.random() * temp + min); 52 } 53 } else 54 { 55 temp = max - x - min + 1;// 只有x>0的情况会出现,正数过小,需减负数 56 y = 0 - (int) (Math.random() * temp) - min; 57 } 58 return y; 59 }加减法的后一位数选定
当有乘除时,则根据上一个数生成操作符,再根据操作符生成合法的后一位数。
1 // 操作符的选定 2 public static int operator(int num, int middle2, int middle3) 3 { 4 if (Math.abs(num) <= middle2) 5 {// 除法下界 6 if (Math.abs(num) < middle3) 7 { 8 return 3; 9 } else 10 { 11 return 0; 12 } 13 } else if (Math.abs(num) >= middle3) 14 {// 乘法上界 15 return 2; 16 } else 17 { 18 return (int) (Math.random() * 4); 19 } 20 } 21 // 下一位数字的选定 22 public static int[] numberB(int key, int num, int lower, int upper) 23 { 24 int[] find = new int[] { 0, lower }; 25 if (key == 0) 26 { 27 find[1] = decide0(num, lower, upper); 28 return find; 29 } else if (key == 2) 30 { 31 int[] judge = new int[2]; 32 judge = decide2(num, lower);// 确保能够整除,并不低于下限 33 if (judge[0] == 0) 34 { 35 find[1] = judge[1]; 36 return find; 37 } else 38 { 39 find[0] = 1; 40 } 41 } else if (key == 3) 42 { 43 find[1] = decide3(num, lower, upper); 44 if (find[0] == 0) 45 { 46 return find; // 乘法不超过上限 47 } 48 } 49 find[1] = decide1(num, lower, upper); 50 return find; 51 }操作符选定以及下一位数字的选定
这样大大减少了重新调用函数的问题,并且实现了运算过程与数值皆在范围内的功能。
在附加题记录用户模块,一开始使用contains(name)函数判断用户,后来发现这样会出现abc与abcabc被认为同一个人而的情况,经过思考,我们使用字符串的断开。
1 String[] arrays = txt.split(" ");
再使用equals(String)函数判断用户,解决了这个问题。
其中,生成有括号与乘除的式子生成的函数判断耗时最多,因为它的判断较多,限制较多,优先级易改变,容易生成最终不合法的式子而重新运行。
1 // 带括号的四则运算 2 public static String Bracket(int lower, int upper, int o) { 3 int middle2 = lower * lower;// 除法下界 4 int middle3 = upper / lower;// 乘法上界 5 int brack_left = 0; // 记录未匹配的左括号个数 6 int brack = 0; // 括号个数 7 int j = 0; 8 char[] p = new char[] { '+', '-', '÷', '*' }; 9 String temp1 = ""; 10 int[] num = new int[o + 1]; // 数字 11 int[] key = new int[o]; // 符号所在的下标 12 num[0] = (int) (Math.random() * (upper - lower + 1) + lower); 13 int result; 14 int[] find = new int[2]; 15 for (j = 0; j < (o - 1); j++) { 16 if (num[j] < 0) { 17 temp1 += "(" + String.valueOf(num[j]) + ")"; 18 } else { 19 temp1 += String.valueOf(num[j]); 20 } 21 int tmpcnt = brack_left; 22 for (int i = 0; i < tmpcnt; i++) { // 若当前有未匹配的左括号,则对每一个未匹配的左括号,都有一定概率生成相应右括号。 23 if ((int) (Math.random() * 5) > 1) { // 生成右括号概率为0.6 24 brack_left--; 25 temp1 += ")"; 26 } 27 } 28 key[j] = calculate.operator(num[j], middle2, middle3); 29 find = calculate.numberB(key[j], num[j], lower, upper); 30 if (find[0] == 1) { 31 key[j] = 1; 32 } 33 num[j + 1] = find[1]; 34 temp1 += String.valueOf(p[key[j]]); 35 if (((brack * 2) <= o) && (((int) (Math.random() * 2)) == 0)) { // 以一定概率生成左括号,概率为1/2 36 temp1 += "("; 37 brack++; 38 brack_left++; 39 j++; 40 if (num[j] < 0) { 41 temp1 += "(" + String.valueOf(num[j]) + ")"; 42 } else { 43 temp1 += String.valueOf(num[j]); 44 } // 生成左括号后必须生成一个数字和运算符,不然可能出现(15)这样的错误 45 key[j] = calculate.operator(num[j], middle2, middle3); 46 find = calculate.numberB(key[j], num[j], lower, upper); 47 if (find[0] == 1) { 48 key[j] = 1; 49 } 50 num[j + 1] = find[1]; 51 temp1 += p[key[j]]; 52 } 53 } 54 while (j != o) { // 判断是否为最后一个数 55 if (num[j] < 0) { 56 temp1 += "(" + String.valueOf(num[j]) + ")"; 57 } else { 58 temp1 += String.valueOf(num[j]); 59 } 60 key[j] = calculate.operator(num[j], middle2, middle3); 61 temp1 += p[key[j]]; 62 find = calculate.numberB(key[j], num[j], lower, upper); 63 if (find[0] == 1) { 64 key[j] = 1; 65 } 66 j++; 67 num[j] = find[1]; 68 } 69 if (num[o] < 0) { 70 temp1 += "(" + String.valueOf(num[o]) + ")"; 71 } else { 72 temp1 += String.valueOf(num[o]); 73 } 74 while ((brack_left) != 0) { // 补全右括号 75 temp1 += ")"; 76 brack_left--; 77 } 78 result = stack.work(temp1, lower, upper, 1); 79 if (result == 0) { 80 temp1 = Bracket(lower, upper, o); 81 } 82 return temp1; 83 84 } 85 86 }有括号四则运算
项目总体分析图,从内存,多线程,CPU等方面分析了计算模块的性能,截图如下:
性能分析过程截图:
按F4,出现以下截图。资源全部被回收。证明没有资源泄露。程序性能良好。
使用单元测试的CPU分析如下图:
使用Command.java的CPU效能分析如下图:
单元测试
1 @Test 2 public void testWork() { 3 assertEquals(0, stack.work("7-5÷(1*37)÷(1*83)", 1, 900, 1)); 4 assertEquals(30, stack.work("55+(-25)÷5*(20-15)", 2, 300, 1)); 5 assertEquals(80, stack.work("((55+25)÷5)*(20-15)", 2, 300, 1)); 6 assertEquals(0, stack.work("60*(20-15)", 2, 200, 1)); 7 }栈的测试
第一个断言测试的是无法整除返回错误标志0;
第二个断言测试的是负数运算;
第三个断言测试的是特殊括号位置的运算;
第四个断言测试的是超过数值返回错误标志0。
1 @Test 2 public void testAll() { 3 // 顺序不同以及异常测试。生成的文件会被覆盖。 4 String[] arg0 = new String[] { "-n", "100", "-m", "5", "100", "-o", "3", "-c", "-b" }; 5 String[] arg1 = new String[] { "-m", "5", "50", "-o", "3", "-n", "100", "-c" }; 6 String[] arg2 = new String[] { "-o", "3", "-m", "5", "50", "-n", "100", "-b" }; 7 String[] arg3 = new String[] { "-n", "100", "-o", "3", "-m", "5", "50" }; 8 Command.main(arg0);// 有括号四则运算测试 9 Command.main(arg1);// 四则运算测试 10 Command.main(arg2);// 有括号加减运算测试 11 Command.main(arg3);// 加减运算测试 12 }命令行正确输入测试
该部分测试的命令行的更改输入顺序的四种出题选择正常运行。输入异常部分请看第七点。
命令行单元测试覆盖率截图如下:
1 @Test 2 public void testDecide2() { 3 int[] find = new int[2]; 4 find = calculate.decide2(20, 2); 5 assertEquals(2, find[1]); 6 find = calculate.decide2(13, 2); 7 assertEquals(1, find[0]); 8 }除法选择除数测试
decide2(int x, int min)为除法选择除数的函数,函数如下:
1 // 被除数能被除数整除并不低于最小 2 public static int[] decide2(int x, int min) 3 { 4 int[] judge = new int[] { 1, 0 }; 5 int temp = Math.abs(x) / min - min + 1;// 除数的范围 6 for (int i = min; i < (temp + min); i++) 7 { 8 if (Math.abs(x) % i == 0) 9 {// 判断是否整除 10 judge[0] = 0; 11 judge[1] = i; 12 return judge; 13 } 14 } 15 return judge; 16 }decide2函数
其中,judge[0]用于判断该数能否有可整除的除数,1为没有,0为有,judge[1]为除数的值。该单元测试则测试了一次可产生除数与一次不能产生除数的情况。
异常说明
1 @Test 2 public void testAll() { 3 String[] arg4 = new String[] { "-o", "3", "-m", "5", "50", "-n" }; 4 String[] arg4_1 = new String[] { "-o", "3", "-n", "-m", "5", "50" }; 5 String[] arg4_2 = new String[] { "-n", "100000", "-m", "5", "50" }; 6 String[] arg4_3 = new String[] { "-o", "3", "-m", "5", "50" }; 7 8 String[] arg5 = new String[] { "-n", "50" }; 9 String[] arg5_1 = new String[] { "-m", "5", "-n", "50", "-o", "3" }; 10 String[] arg5_2 = new String[] { "-n", "50", "-m", "3" }; 11 String[] arg5_3 = new String[] { "-n", "50", "-o", "3", "-m" }; 12 String[] arg5_4 = new String[] { "-m", "-n", "50" }; 13 14 String[] arg6 = new String[] { "-o", "11", "-m", "5", "50", "-n", "100" }; 15 String[] arg6_1 = new String[] { "-n", "100", "-o", "-m", "5", "50" }; 16 String[] arg6_2 = new String[] { "-n", "100", "-m", "5", "50", "-o" }; 17 18 String[] arg7 = new String[] { "-m", "5", "20", "-n", "100", "-c" }; 19 String[] arg7_1 = new String[] { "-m", "5", "50", "-n", "100", "-b" }; 20 21 String[] arg8 = new String[] { "-b", "1", "-o", "3", "-m", "5", "50", "-n", "100" }; 22 String[] arg8_1 = new String[] { "-c", "1", "-o", "3", "-m", "5", "50", "-n", "100" }; 23 String[] arg8_2 = new String[] { "-n", "100", "-m", "5", "50", "-d" }; 24 25 Command.main(arg4);// 缺少题数值测试 26 Command.main(arg4_1); 27 Command.main(arg4_2);// 题数值过大测试 28 Command.main(arg4_3);// 缺少题数测试 29 30 Command.main(arg5);// 缺少数值范围 31 Command.main(arg5_1);// 缺少数值范围上限测试 32 Command.main(arg5_2); 33 Command.main(arg5_3);// 缺少数值范围上下限测试 34 Command.main(arg5_4); 35 36 Command.main(arg6);// 操作符数值过大测试 37 Command.main(arg6_1);// 缺少操作符数值测试 38 Command.main(arg6_2); 39 40 Command.main(arg7);// 乘除需要上界大于下界的平方 41 Command.main(arg7_1);// 括号需要操作符数大于1 42 43 Command.main(arg8);// 输入非法测试之b后有数字 44 Command.main(arg8_1);// 输入非法测试之c后有数字 45 Command.main(arg8_2);// 输入非法测试之无辨识字符 46 }命令行异常输入测试
对于命令行可能出现的异常大概有13个:
- 缺少题数值(-n后无带数字,如arg4与arg4_1)时,提醒缺少题数值,并告知-n的范围;
- 题数值过大(-n后数值超过10000,如arg4_2)时,提醒告知题数值范围(过小同理);
- 缺少题数(命令中无-n,如arg4_3)时,提醒-n为必须项,并告知-n范围。
- 缺少数值范围(命令中无-m,如arg5)时,提醒-m为必须项,并告知-m上下限各自范围;
- 缺少数值范围上限(-m后只带一个数字,如arg5_1和 arg5_2)时,提醒缺少上限,并告知上限范围;
- 缺少数值范围上下限(-m后不带数字,如arg5_3和 arg5_4)时,提醒缺少上下限,并告知上下限各自范围;
- 数值范围数值过小过大时,提醒告知操作符数值范围。
- 操作符数值过大(-o后数值超过10,如arg6)时,提醒告知操作符数值范围(过小同理);
- 缺少操作符数值(输入-o,后方没有带数值,如arg6_1与arg6_2)时,提醒缺少操作符数值,并告知-o范围。
- 选择乘除法但是上界小于下界的平方,无法生成含有乘除的式子(如arg7)时,提醒上界需大于下界的平方;
- 选择括号但是操作符默认为1或选择为1,不符合生成括号的条件(如arg7_1)时,提醒选择括号需要操作符数大于1。
- –b(或-c)后带数字(如arg8与arg8_1),提醒-b(或-c)后不能带数字;
- 出现除m、n、o、b、c外的字符如d等(如arg8_2),提醒输入值非法。
界面模块的详细设计过程
设计图如下:
我们先从选择出题或做题开始。
选择出题则进入出题参数输入界面。
利用MouseListener的mouseEntered(MouseEvent e)与setTitle(String);使得鼠标移到参数上,标题会有提示功能。
输入完毕点击确认后,由输入的参数判断是否有异常并提示直至无异常创建文件。
1 public class submitListener implements ActionListener { 2 public void actionPerformed(ActionEvent e) { 3 String m = "题数与数值上下限为必填项,请按标题提示输入正整数!"; 4 String m2 = "创建文件成功!"; 5 int n0, lower0, upper0, o0, c0, b0; 6 o0 = 1; 7 c0 = 0; 8 b0 = 0; 9 String o1 = ""; 10 try { 11 n0 = Integer.parseInt(n.getText()); 12 lower0 = Integer.parseInt(lower.getText()); 13 upper0 = Integer.parseInt(upper.getText()); 14 if (n0 < 1 || n0 > 10000) { 15 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), "题数范围为1-10000", "提示", 16 JOptionPane.INFORMATION_MESSAGE); 17 return; 18 } 19 if (lower0 < 1 || lower0 > 100) { 20 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), "下界范围为1-100", "提示", 21 JOptionPane.INFORMATION_MESSAGE); 22 return; 23 } 24 if (upper0 < 50 || upper0 > 1000) { 25 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), "上界范围为50-1000", "提示", 26 JOptionPane.INFORMATION_MESSAGE); 27 return; 28 } 29 if (upper0 < (2 * lower0)) { 30 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), "上界必须大于两倍下界", "提示", 31 JOptionPane.INFORMATION_MESSAGE); 32 return; 33 } 34 } catch (NumberFormatException e2) { 35 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), m, "提示", JOptionPane.INFORMATION_MESSAGE); 36 return; 37 } 38 try { 39 o1 = o.getText(); 40 o0 = Integer.parseInt(o1); 41 } catch (NumberFormatException e2) { 42 if (!o1.equals("")) { 43 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), "请输入1-10的正整数或不输入保持默认,默认为1", "提示", 44 JOptionPane.INFORMATION_MESSAGE); 45 return; 46 } 47 } 48 if (c.isSelected()) { 49 c0 = 1; 50 } 51 if (b.isSelected()) { 52 b0 = 1; 53 } 54 if (o0 == 1 && b0 == 1) { 55 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), "括号需要操作符数量大于1", "提示", 56 JOptionPane.INFORMATION_MESSAGE); 57 return; 58 } 59 if (c0 == 1 && upper0 < (lower0 * lower0)) { 60 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), "乘除法需要上界数值大于下界的平方", "提示", 61 JOptionPane.INFORMATION_MESSAGE); 62 return; 63 } 64 createFile.fileCreate(n0, lower0, upper0, o0, c0, b0); 65 JOptionPane.showMessageDialog(JOptionPane.getRootFrame(), m2, "提示", JOptionPane.INFORMATION_MESSAGE); 66 System.exit(0); 67 } 68 }参数确认
选择做题则先输入做题人名字(在这里建议使用英文,中文名字无法很好的记录)。
接着上传文件,在这里使用了txt文件过滤器,使之仅可上传txt文件。
1 FileFilter filter = new FileNameExtensionFilter("Text file", "txt"); 2 JFileChooser fileChooser = new JFileChooser(); 3 fileChooser.setAcceptAllFileFilterUsed(false); 4 fileChooser.addChoosableFileFilter(filter); 5 FileSystemView fsv = FileSystemView.getFileSystemView();过滤器
另外,出题与做题都统一为utf-8编码,免去执行文件编码错误。
BufferedWriter bw = new BufferedWriter(new OutputStreamWriter(new FileOutputStream(file, true), "UTF-8"));
InputStreamReader read = new InputStreamReader(new FileInputStream(file), "utf-8");
上传成功后开始计时做题,并于最后结果中显示用时。
经过后面JTextPane控件的启发,我考虑到出题时题目长度有长短,为了更加美观显示,应该需要自动换行,我同样采用了HTML编辑文本的想法,做出改进。
1 public static void JlabelSetText(JLabel jLabel, String longString) throws InterruptedException { 2 StringBuilder builder = new StringBuilder("<html>"); 3 char[] chars = longString.toCharArray(); 4 FontMetrics fontMetrics = jLabel.getFontMetrics(jLabel.getFont()); 5 int start = 0; 6 int len = 0; 7 while (start + len < longString.length()) { 8 while (true) { 9 len++; 10 if (start + len > longString.length()) 11 break; 12 if (fontMetrics.charsWidth(chars, start, len) > jLabel.getWidth()) { 13 break; 14 } 15 } 16 builder.append(chars, start, len - 1).append("<br/>"); 17 start = start + len - 1; 18 len = 0; 19 } 20 builder.append(chars, start, longString.length() - start); 21 builder.append("</html>"); 22 jLabel.setText(builder.toString()); 23 }换行
计时方面我原本采用秒数计时,后来考虑到当做题时间较长时,秒数很难清晰明确的表达,所以改用了 hh:mm:ss 法显示。
1 public static String getTimeStrBySecond(long second) { 2 if (second <= 0) { 3 return "00:00:00"; 4 } 5 int hours = (int) second / HOUR_SECOND; 6 if (hours > 0) { 7 8 second -= hours * HOUR_SECOND; 9 } 10 int minutes = (int) second / MINUTE_SECOND; 11 if (minutes > 0) { 12 second -= minutes * MINUTE_SECOND; 13 } 14 return (hours > 10 ? (hours + "") 15 : ("0" + hours) + ":" + (minutes > 10 ? (minutes + "") : ("0" + minutes)) + ":" 16 + (second > 10 ? (second + "") : ("0" + second))); 17 }秒数转换
自动换行处理与秒数转换被我写入新类——dataDeal类中。
最终做完题目后除了显示用时,还显示题数、分数、错题以及该题正确答案,非首次用户会显示历史分数以及最高分数。
原本该部分使用了JTextArea控件,但学姐建议正确答案部分对齐显示会更加美观,并提出了C#中的ListView控件,但很遗憾,Java中似乎并没有。JTextArea控件是纯文本显示,很难做到不同的对齐方式,所以我删除了该类。经过多方学习比较,我最终选择了JTextPane控件,该控件简单易用,可将文本显示为HTML文本,大大提高了编辑的样式性。我最终采取了表格法对齐,另外对重点突出的地方加粗变红显示,达到强调与一定视觉冲击效果,可从后文看到对比图。
1 String text = "<p style='font-family:楷体; font-size:19'>" + name + " 本次用时<span style='color:red'><strong> " 2 + dataDeal.getTimeStrBySecond(spentTime) + " </strong></span>,得分<span style='color:red'><strong> " 3 + goal + " </strong></span>分。<br>"; 4 if (size0 == 0) { 5 text += "你总共答了<span style='color:red'><strong> " + size 6 + " </strong></span>道题,并全部答对!<span style='color:red'><strong>恭喜!</strong></span></p>"; 7 } else { 8 text += "你总共答了<span style='color:red'><strong> " + size 9 + " </strong></span>道题,答对<span style='color:red'><strong> " + size1 10 + " </strong></span>道,答错<span style='color:red'><strong> " + size0 11 + " </strong></span>道,分别为:</p><p><table border=0>"; 12 for (int i = 0; i < (size0 * 2); i++) { 13 text += "<tr><td style='font-family:楷体; font-size:19'><strong>" + wrong.get(i++) 14 + " </strong></td><td width='180' style='font-family:楷体; font-size:19;color:red'><strong> " 15 + wrong.get(i) + "</strong></td></tr>"; 16 } 17 } 18 text += "</table></p>"; 19 text += "<p style='font-family:楷体; font-size:19'>" + createFile.record(name, goal) + "</p>"; 20 21 JTextPane textarea = new JTextPane(); 22 textarea.setContentType("text/html"); 23 textarea.setText(text); 24 textarea.setEditable(false); 25 JScrollPane textAreascrollPane = new JScrollPane(textarea); 26 add(textAreascrollPane, BorderLayout.CENTER);JTextPane
界面模块与计算模块的对接
如图所示
在界面模块选择出题输入参数之后调用fileCreate类,再由fileCreate类调用计算模块,创建result.txt
在界面模块选择做题输入名字、上传文件、做题。做题时调用计算模块的stack类计算判断正确性,记录错题。最终结果由计算模块中的fileCreate类的record(String name, int goal)记录,由界面模块显示。
实现的功能大致有12个,并且为了提高用户体验,修改了图标并增加了背景,将操作符数修改为下拉框选择,默认选择为1,避免输入非数字错误:
模式选择
出题参数输入(前后对比图)
出题参数要求提醒
输入参数有误提醒(见第七点异常)
生成文件
记录用户
上传文件(只允许txt文件)
判断文件是否为空或非练习题
计时
一道一道做题并且题目过长时自动换行
评分
根据学姐给的建议做出了修改,以下为前后对比图,正确答案对齐,使之更加美观。另外我修改了做题时间的显示形式,这样当做题时间较长时可以更加清晰的看出时间情况。而做题时间、得分情况、错题与正确答案皆加粗甚至标红,使之更加显眼,提高用户体验。
记录历史分数与最高分数
结对编程
我们先一起分析了需求与功能的实现,并提出了一些有实质性的方法,并确认数据的传递方式。再分析各自的个人项目代码,指出了双方优劣性,在综合考虑选择基础代码加以改进。
我们根据自己较为擅长的方面分工,如相对之下,我对gui较为熟悉,而她对字符串处理较为熟悉,则我负责界面展示而她负责命令行的分析。各自写完之后我们再复审双方代码,对代码不理解之处询问并补充注释,以及对双方异常情况补充。最后在一起整合双方代码,使之成为完整项目。
结对编程的优缺点
在此过程中我们互相帮助、互相学习、能力上得到互补,而代码和产品质量提高,有效地减少bug并且考虑到更多方面的情况。有两台电脑可以测试程序效果,如她的电脑比我小,我的gui显示不同,她的部分算式被遮挡,最终我选择了将按钮部分的面板设为透明,解决了这个问题。
不足之处在于队友之间的进度相互影响,不同的代码风格之间的磨合也花费了一定时间。
双方优缺点:
庄莉 | 王璐瑶 | |
优点 | 认真细心,有责任心 | 任劳任怨 |
代码能力高 | 对字符串以及字符串数组的处理十分熟练 | |
动手能力强 | 很有想法,有好点子 | |
缺点 | 有时候对于小问题过于钻牛角尖 | 因生病而不在状态,没注意到比较细的地方,时间较少 |
实际PSP
PSP | 任务内容 | 实际完成需要的时间(min) |
Planning | 计划 | 0.5 |
Estimate | 估计这个任务需要多少时间,并规划大致工作步骤 | 0.5 |
Development | 开发 | 53.25 |
Analysis | 需求分析 (包括学习新技术) | 0.5 |
Design Spec | 生成设计文档 | 0.25 |
Design Review | 设计复审 (和同事审核设计文档) | 0.25 |
Coding Standard | 代码规范 (为目前的开发制定合适的规范) | 0.25 |
Design | 具体设计 | 1 |
Coding | 具体编码 | 40 |
Code Review | 代码复审 | 1 |
Test | 测试(自我测试,修改代码,提交修改) | 10 |
Reporting | 报告 | 9 |
Test Report | 测试报告 | 8 |
Size Measurement | 计算工作量 | 0.5 |
Postmortem & Process Improvement Plan | 事后总结, 并提出过程改进计划 | 0.5 |
本次结对编程真的让我学到很多知识,尤其是各种操作,就像上一篇博客的链接一样,我查了许许多多这样的链接,学习了一种有一种的方法,与队友配合,完成了这次项目。而每次写博客,都能重新总结我的思路,受益良多。
虽然真的很辛苦,但能做出来也就够了。
以下部分由于时间与精力关系,我们小组并没有完成,仅提供思路参考,有想法的同学可加以尝试。
附加题多语言思路参考:程序国际化
思路注意点参考:
- 文字描述使用短语、名词与阿拉伯数字,便于翻译。 标准语言建议英语,中文字符难以使用字符串判断内容。
- 语言选择可为下拉框列表旁边带一不可编辑的文本框。由于可以自添加,所以可以建立一个文件(如txt),文件中加入选项,执行代码时读出文件,for循环将每一项添加进下拉框列表中。有一项设置为其他或自定义,选择它时,旁边的文本框变为可编辑,用于添加语言(也可让下拉框为可编辑,当输入的语言下拉选项中没有时认为添加新语言)。
- 添加语言则需配置文件(参考网址),将所有需要翻译的文字让用户对应翻译。生成新的配置文件,并命名为该语言(使用英文命名),将该语言添加到保存下拉框选项的文件中。
- 当用户选择一种语言时,通过其相同的命名,即可调用该语言的配置文件进行翻译,达到多语言转化功能。
转载于:.html
发布评论