- 笔者:万宇龙
- 项目作者:梁钰
项目目标
个人项目:中小学数学卷子自动生成程序
用户:小学、初中和高中数学老师。
功能:
- 命令行输入用户名和密码,两者之间用空格隔开(程序预设小学、初中和高中各三个账号,具体见附表),如果用户名和密码都正确,将根据账户类型显示“当前选择为XX出题”,XX为小学、初中和高中三个选项中的一个。否则提示“请输入正确的用户名、密码”,重新输入用户名、密码;
- 登录后,系统提示“准备生成XX数学题目,请输入生成题目数量(输入-1将退出当前用户,重新登录):”,XX为小学、初中和高中三个选项中的一个,用户输入所需出的卷子的题目数量,系统默认将根据账号类型进行出题。每道题目的操作数在1-5个之间,操作数取值范围为1-100;
- 题目数量的有效输入范围是“10-30”(含10,30,或-1退出登录),程序根据输入的题目数量生成符合小学、初中和高中难度的题目的卷子(具体要求见附表)。同一个老师的卷子中的题目不能与以前的已生成的卷子中的题目重复(以指定文件夹下存在的文件为准,见5);
- 在登录状态下,如果用户需要切换类型选项,命令行输入“切换为XX”,XX为小学、初中和高中三个选项中的一个,输入项不符合要求时,程序控制台提示“请输入小学、初中和高中三个选项中的一个”;输入正确后,显示“”系统提示“准备生成XX数学题目,请输入生成题目数量”,用户输入所需出的卷子的题目数量,系统新设置的类型进行出题;
- 生成的题目将以“年-月-日-时-分-秒.txt”的形式保存,每个账号一个文件夹。每道题目有题号,每题之间空一行;
- 个人项目9月17日晚上10点以前提交至创新课程管理系统。提交方式:工程文件打包,压缩包名为“几班+姓名.rar”。迟交2天及以内者扣分,每天扣20%。迟交2天及以上者0分。
附表-1:账户、密码
账户类型 | 账户 | 密码 | 备注 |
---|---|---|---|
小学 | 张三1 | 123 | |
小学 | 张三2 | 123 | |
小学 | 张三3 | 123 | |
初中 | 李四1 | 123 | |
初中 | 李四2 | 123 | |
初中 | 李四3 | 123 | |
高中 | 王五1 | 123 | |
高中 | 王五2 | 123 | |
高中 | 王五3 | 123 |
附表-2:小学、初中、高中题目难度要求
小学 | 初中 | 高中 | |
---|---|---|---|
难度要求 | +,-,*,/ | 平方,开根号 | sin,cos,tan |
备注 | 只能有+,-,*,/和() | 题目中至少有一个平方或开根号的运算符 | 题目中至少有一个sin,cos,tan的运算符 |
项目设计
项目分为了4个包:entity
、service
、test
、utils
-
entity
包含了构成整个代码生成器部分的实体,类如下
Exam
、JuniorExam
、PrimaryExam
、SeniorExam
、User
-
service
向外提供项目服务,类如下
MyService
-
test
测试用的包,不做说明 -
utils
存储一些工具类,为上面的enetity
包服务
项目实现
-
最重要的部分:题目生成是通过字符串拼接形式完成
首先生成n个操作数,接着是n-1个操作符,插在n个操作数之间
同时生成一个长度为n的boolean类型数组,用来加入高级运算
最后随机产生括号
下面的类都是基于这个思想进行编写 -
entity
抽象Exam
将+,-,*,/作为基本二元运算符存储;声明了最大操作数个数;声明了存储操作数的数组
PrimaryExam
、JuniorExam
、SeniorExam
都继承自Exam
,并且都有public static String createExam()
方法(根据不同的学校通过字符串拼接创建不同的题目)。
User
拥有三个属性:level
、id
、password
以及对应的Getter
和Setter
-
service
MyService
由于不要求注册或者是新增用户,直接使用ArrayList进行初始化和记录所有用户
管理entity
里面的类,让其相互协同工作,相互配合,以达到登录以及出题的效果 -
utils
ChoiceUtil
判断的工具类,有可以判断字符串是否为纯数字的方法
ExamProduceUtils
试卷生成工具类,可以保存、读取试卷,实现查重功能
项目测试
正常登录
能防止异常输入登录
能防止输入异常题目数量
正常切换模式
正常出题
小学:
初中:
高中:
能防止出题界面异常输入
项目质量
-
可读性
可读性很高,有足够量的注释,变量、方法以及类的命名一目了然 -
可拓展性
有点遗憾的就是这里了,如果想要新增加一个一元或者二元运算符,那么需要改动原代码,这可能导致更多的工作以及无法预测的问题 -
灵活性
由于整个项目被拆解成很多个部分,再由各个部分合作完成,因此有较高灵活性 -
编码规范
-
- 关于Javadoc,需要按照
@param
,@return
,@throws
,@deprecated
的形式出现,顺便再这样写会更好看一点
- 关于Javadoc,需要按照
/**
* 描述内容1
* 描述内容2
* (有一行空出来什么都不要写,包括这句话)
* @param 变量1 变量1描述
* @param 变量2 变量2描述
* @return 返回值描述
*/
-
- 关于
{}
大括号与if
,else
,for
,do
,while
语句一起使用,即使只有一条语句(或是空),也应该把大括号{}
写上。
- 关于
-
- 关于自动换行
如果在 非赋值运算符 处断开,那么在该符号前断开(比如+,它将位于下一行)。(有一个toString()
方法中的+
放在了一行的最后)
如果在 赋值运算符 处断开,通常的做法是在该符号后断开(比如=,它与前面的内容留在同一行)。
- 关于自动换行
-
生成题目
没有做计算答案部分,不方便后续拓展,没有合理性的判断,可能出现÷0,根号下负数,tan90的情况。运算符难以嵌套(不会出现\(tan(cos(\frac{\sqrt{a^2+b^2}}{a}))\)类似的式子)