diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..f68d109 --- /dev/null +++ b/.gitignore @@ -0,0 +1,29 @@ +### IntelliJ IDEA ### +out/ +!**/src/main/**/out/ +!**/src/test/**/out/ + +### Eclipse ### +.apt_generated +.classpath +.factorypath +.project +.settings +.springBeans +.sts4-cache +bin/ +!**/src/main/**/bin/ +!**/src/test/**/bin/ + +### NetBeans ### +/nbproject/private/ +/nbbuild/ +/dist/ +/nbdist/ +/.nb-gradle/ + +### VS Code ### +.vscode/ + +### Mac OS ### +.DS_Store \ No newline at end of file diff --git a/.idea/.gitignore b/.idea/.gitignore new file mode 100644 index 0000000..10b731c --- /dev/null +++ b/.idea/.gitignore @@ -0,0 +1,5 @@ +# 默认忽略的文件 +/shelf/ +/workspace.xml +# 基于编辑器的 HTTP 客户端请求 +/httpRequests/ diff --git a/.idea/.name b/.idea/.name new file mode 100644 index 0000000..2b3e571 --- /dev/null +++ b/.idea/.name @@ -0,0 +1 @@ +untitled111.iml \ No newline at end of file diff --git a/.idea/artifacts/untitled111_jar.xml b/.idea/artifacts/untitled111_jar.xml new file mode 100644 index 0000000..5ef401c --- /dev/null +++ b/.idea/artifacts/untitled111_jar.xml @@ -0,0 +1,21 @@ + + + $PROJECT_DIR$/out/artifacts/untitled111_jar + + + + + + + + + + + + + + + + + + \ No newline at end of file diff --git a/.idea/misc.xml b/.idea/misc.xml new file mode 100644 index 0000000..2bfdeda --- /dev/null +++ b/.idea/misc.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/.idea/modules.xml b/.idea/modules.xml new file mode 100644 index 0000000..5116208 --- /dev/null +++ b/.idea/modules.xml @@ -0,0 +1,8 @@ + + + + + + + + \ No newline at end of file diff --git a/.idea/vcs.xml b/.idea/vcs.xml new file mode 100644 index 0000000..94a25f7 --- /dev/null +++ b/.idea/vcs.xml @@ -0,0 +1,6 @@ + + + + + + \ No newline at end of file diff --git a/src/ExamManager.java b/src/ExamManager.java new file mode 100644 index 0000000..986ecfc --- /dev/null +++ b/src/ExamManager.java @@ -0,0 +1,24 @@ +import java.util.*; + +public class ExamManager { + private QuestionGenerator generator = new QuestionGenerator(); + private Set history = new HashSet<>(); + private String type; + + public ExamManager(String type){ + this.type = type; + } + + public List generateExam(int numQuestions){ + List questions = new ArrayList<>(); + for(int i=0;i 密码 + private Map userDatabase = new HashMap<>(); + // 存储用户名 -> 邮箱 映射 + private Map usernameToEmail = new HashMap<>(); + // 存储邮箱 -> 用户名 映射 + private Map emailToUsername = new HashMap<>(); + + private Map currentUser = new HashMap<>(); // 存储当前登录用户信息 + private List currentExam = new ArrayList<>(); + private int currentIndex = 0; + private int score = 0; + private ExamManager examManager; + private QuestionGenerator questionGenerator = new QuestionGenerator(); + + public static void main(String[] args) { + launch(args); + } + + @Override + public void start(Stage stage) { + primaryStage = stage; + primaryStage.setTitle("小初高数学学习软件"); + primaryStage.setScene(buildLoginScene()); + primaryStage.show(); + } + + // 登录界面 + private Scene buildLoginScene() { + VBox root = new VBox(10); + root.setStyle("-fx-padding: 20;"); + Label titleLabel = new Label("数学学习软件 - 登录"); + titleLabel.setStyle("-fx-font-size: 16; -fx-font-weight: bold;"); + + TextField loginField = new TextField(); + loginField.setPromptText("请输入邮箱或用户名"); + PasswordField passwordField = new PasswordField(); + passwordField.setPromptText("请输入密码"); + + Button loginBtn = new Button("登录"); + Button registerBtn = new Button("前往注册"); + Label infoLabel = new Label(); + Label tipLabel = new Label("提示:可以使用邮箱或用户名登录"); + tipLabel.setStyle("-fx-font-size: 10; -fx-text-fill: gray;"); + + // 登录按钮事件 + loginBtn.setOnAction(e -> { + String loginInput = loginField.getText().trim(); + String password = passwordField.getText(); + + if (loginInput.isEmpty() || password.isEmpty()) { + infoLabel.setText("请输入登录信息和密码"); + return; + } + + String email = getEmailFromLoginInput(loginInput); + if (email == null) { + infoLabel.setText("用户不存在"); + return; + } + + if (!userDatabase.get(email).equals(password)) { + infoLabel.setText("密码错误"); + return; + } + + // 登录成功 + currentUser.put("email", email); + currentUser.put("username", emailToUsername.get(email)); + infoLabel.setText("登录成功!欢迎 " + emailToUsername.get(email)); + primaryStage.setScene(buildLevelSelectionScene()); + }); + + // 注册按钮事件 + registerBtn.setOnAction(e -> { + primaryStage.setScene(buildRegisterScene()); + }); + + root.getChildren().addAll(titleLabel, loginField, passwordField, tipLabel, loginBtn, registerBtn, infoLabel); + return new Scene(root, 400, 300); + } + + // 根据登录输入获取邮箱 + private String getEmailFromLoginInput(String loginInput) { + // 如果输入包含@,认为是邮箱 + if (loginInput.contains("@")) { + return userDatabase.containsKey(loginInput) ? loginInput : null; + } else { + // 否则认为是用户名 + return usernameToEmail.get(loginInput); + } + } + + // 注册界面 + private Scene buildRegisterScene() { + VBox root = new VBox(10); + root.setStyle("-fx-padding: 20;"); + Label titleLabel = new Label("用户注册"); + titleLabel.setStyle("-fx-font-size: 16; -fx-font-weight: bold;"); + + TextField usernameField = new TextField(); + usernameField.setPromptText("设置用户名 (4-10位字符)"); + TextField emailField = new TextField(); + emailField.setPromptText("请输入邮箱"); + Button sendCodeBtn = new Button("发送验证码"); + TextField codeField = new TextField(); + codeField.setPromptText("请输入验证码"); + PasswordField passwordField = new PasswordField(); + passwordField.setPromptText("设置密码 (6-10位,含大小写字母和数字)"); + PasswordField confirmPasswordField = new PasswordField(); + confirmPasswordField.setPromptText("确认密码"); + Button registerBtn = new Button("注册"); + Button backBtn = new Button("返回登录"); + Label infoLabel = new Label(); + + sendCodeBtn.setOnAction(e -> { + String username = usernameField.getText().trim(); + String email = emailField.getText().trim(); + + // 验证用户名 + if (!isValidUsername(username)) { + infoLabel.setText("用户名必须4-10位,只能包含字母、数字和下划线"); + return; + } + + if (usernameToEmail.containsKey(username)) { + infoLabel.setText("用户名已存在"); + return; + } +/* + // 使用新的邮箱验证方法 + if (!isValidEmail(email)) { + infoLabel.setText("邮箱格式不正确,请使用有效的邮箱地址"); + return; + } +*/ + if (userDatabase.containsKey(email)) { + infoLabel.setText("该邮箱已注册"); + return; + } + + generatedCode = String.format("%06d", new Random().nextInt(999999)); + infoLabel.setText("验证码发送中..."); + + new Thread(() -> { + boolean success = mailSender.sendMail(email, "注册验证码", "你的验证码为:" + generatedCode); + Platform.runLater(() -> { + if (success) { + infoLabel.setText("验证码已发送,请查看邮箱"); + } else { + infoLabel.setText("发送失败,请检查邮箱是否正确"); + } + }); + }).start(); + }); + + registerBtn.setOnAction(e -> { + String username = usernameField.getText().trim(); + String email = emailField.getText().trim(); + String inputCode = codeField.getText().trim(); + String pw = passwordField.getText(); + String pwConfirm = confirmPasswordField.getText(); + + if (!isValidUsername(username)) { + infoLabel.setText("用户名必须4-10位,只能包含字母、数字和下划线"); + return; + } + + if (usernameToEmail.containsKey(username)) { + infoLabel.setText("用户名已存在"); + return; + } + + if (!inputCode.equals(generatedCode)) { + infoLabel.setText("验证码错误"); + } else if (!pw.equals(pwConfirm)) { + infoLabel.setText("两次密码不一致"); + } else if (!isValidPassword(pw)) { + infoLabel.setText("密码必须6-10位,包含大小写字母和数字"); + } else if (userDatabase.containsKey(email)) { + infoLabel.setText("邮箱已注册"); + } else { + // 注册用户 + userDatabase.put(email, pw); + usernameToEmail.put(username, email); + emailToUsername.put(email, username); + infoLabel.setText("注册成功!用户名: " + username); + // 注册成功后自动登录 + currentUser.put("email", email); + currentUser.put("username", username); + primaryStage.setScene(buildLevelSelectionScene()); + } + }); + + backBtn.setOnAction(e -> { + primaryStage.setScene(buildLoginScene()); + }); + + root.getChildren().addAll(titleLabel, usernameField, emailField, sendCodeBtn, codeField, + passwordField, confirmPasswordField, registerBtn, backBtn, infoLabel); + return new Scene(root, 400, 450); + } + + // 用户名验证方法 + private boolean isValidUsername(String username) { + if (username.length() < 4 || username.length() > 10) { + return false; + } + // 只能包含字母、数字、下划线 + return username.matches("^[a-zA-Z0-9_]+$"); + } + + // 密码验证方法 + private boolean isValidPassword(String password) { + if (password.length() < 6 || password.length() > 10) { + return false; + } + boolean hasUpper = false; + boolean hasLower = false; + boolean hasDigit = false; + + for (char c : password.toCharArray()) { + if (Character.isUpperCase(c)) hasUpper = true; + if (Character.isLowerCase(c)) hasLower = true; + if (Character.isDigit(c)) hasDigit = true; + } + + return hasUpper && hasLower && hasDigit; + } + + + + // 难度选择界面 + private Scene buildLevelSelectionScene() { + VBox root = new VBox(15); + root.setStyle("-fx-padding: 20;"); + + // 显示欢迎信息 + String username = currentUser.get("username"); + Label welcomeLabel = new Label("欢迎, " + username + "!"); + welcomeLabel.setStyle("-fx-font-size: 16; -fx-font-weight: bold; -fx-text-fill: #2E8B57;"); + + Label label = new Label("请选择难度"); + label.setStyle("-fx-font-size: 14; -fx-font-weight: bold;"); + + Button primaryBtn = new Button("小学"); + primaryBtn.setStyle("-fx-font-size: 12; -fx-pref-width: 120;"); + Button juniorBtn = new Button("初中"); + juniorBtn.setStyle("-fx-font-size: 12; -fx-pref-width: 120;"); + Button seniorBtn = new Button("高中"); + seniorBtn.setStyle("-fx-font-size: 12; -fx-pref-width: 120;"); + + Button changePwdBtn = new Button("修改密码"); + changePwdBtn.setStyle("-fx-font-size: 12;"); + Button logoutBtn = new Button("退出登录"); + logoutBtn.setStyle("-fx-font-size: 12;"); + + primaryBtn.setOnAction(e -> buildExamNumberScene("小学")); + juniorBtn.setOnAction(e -> buildExamNumberScene("初中")); + seniorBtn.setOnAction(e -> buildExamNumberScene("高中")); + + changePwdBtn.setOnAction(e -> primaryStage.setScene(buildChangePasswordScene())); + logoutBtn.setOnAction(e -> { + currentUser.clear(); + primaryStage.setScene(buildLoginScene()); + }); + + VBox buttonBox = new VBox(10, primaryBtn, juniorBtn, seniorBtn); + buttonBox.setStyle("-fx-alignment: center;"); + + HBox bottomBox = new HBox(10, changePwdBtn, logoutBtn); + bottomBox.setStyle("-fx-alignment: center;"); + + root.getChildren().addAll(welcomeLabel, label, buttonBox, bottomBox); + Scene scene = new Scene(root, 400, 300); + primaryStage.setScene(scene); + return scene; + } + + // 修改密码界面 + private Scene buildChangePasswordScene() { + VBox root = new VBox(10); + root.setStyle("-fx-padding: 20;"); + Label titleLabel = new Label("修改密码"); + titleLabel.setStyle("-fx-font-size: 16; -fx-font-weight: bold;"); + + Label userLabel = new Label("用户: " + currentUser.get("username")); + userLabel.setStyle("-fx-font-size: 12; -fx-text-fill: gray;"); + + PasswordField oldPwdField = new PasswordField(); + oldPwdField.setPromptText("请输入原密码"); + PasswordField newPwdField = new PasswordField(); + newPwdField.setPromptText("请输入新密码"); + PasswordField confirmPwdField = new PasswordField(); + confirmPwdField.setPromptText("确认新密码"); + + Button confirmBtn = new Button("确认修改"); + Button backBtn = new Button("返回"); + Label infoLabel = new Label(); + + confirmBtn.setOnAction(e -> { + String oldPwd = oldPwdField.getText(); + String newPwd = newPwdField.getText(); + String confirmPwd = confirmPwdField.getText(); + String email = currentUser.get("email"); + + if (!userDatabase.get(email).equals(oldPwd)) { + infoLabel.setText("原密码错误"); + } else if (!newPwd.equals(confirmPwd)) { + infoLabel.setText("两次输入的新密码不一致"); + } else if (!isValidPassword(newPwd)) { + infoLabel.setText("新密码必须6-10位,包含大小写字母和数字"); + } else { + userDatabase.put(email, newPwd); + infoLabel.setText("密码修改成功!"); + } + }); + + backBtn.setOnAction(e -> { + primaryStage.setScene(buildLevelSelectionScene()); + }); + + root.getChildren().addAll(titleLabel, userLabel, oldPwdField, newPwdField, confirmPwdField, + confirmBtn, backBtn, infoLabel); + return new Scene(root, 400, 350); + } + + // 输入题目数量界面 + private void buildExamNumberScene(String level) { + VBox root = new VBox(10); + root.setStyle("-fx-padding: 20;"); + Label label = new Label("请输入" + level + "题目数量 (10-30)"); + label.setStyle("-fx-font-size: 14;"); + + TextField numberField = new TextField(); + numberField.setPromptText("题目数量"); + Button startBtn = new Button("开始做题"); + Button backBtn = new Button("返回"); + Label infoLabel = new Label(); + + startBtn.setOnAction(e -> { + try { + int count = Integer.parseInt(numberField.getText().trim()); + if (count < 10 || count > 30) { + infoLabel.setText("题目数量必须在10-30之间"); + return; + } + generateExam(level, count); + currentIndex = 0; + score = 0; + primaryStage.setScene(buildExamScene()); + } catch (NumberFormatException ex) { + infoLabel.setText("请输入有效数字"); + } + }); + + backBtn.setOnAction(e -> { + primaryStage.setScene(buildLevelSelectionScene()); + }); + + root.getChildren().addAll(label, numberField, startBtn, backBtn, infoLabel); + primaryStage.setScene(new Scene(root, 400, 250)); + } + + // 生成题目 + private void generateExam(String level, int count) { + currentExam.clear(); + examManager = new ExamManager(level); + List questionTexts = examManager.generateExam(count); + + for (int i = 0; i < questionTexts.size(); i++) { + String questionText = (i + 1) + ". " + questionTexts.get(i) + " = ?"; + String correctAnswer = questionGenerator.calculateAnswer(questionTexts.get(i)); + + List options = questionGenerator.generateOptions(correctAnswer); + currentExam.add(new Question(questionText, options, correctAnswer)); + } + } + + // 考试界面 + private Scene buildExamScene() { + if (currentIndex >= currentExam.size()) { + return buildScoreScene(); + } + + Question q = currentExam.get(currentIndex); + VBox root = new VBox(15); + root.setStyle("-fx-padding: 20;"); + + Label progressLabel = new Label("进度: " + (currentIndex + 1) + "/" + currentExam.size()); + progressLabel.setStyle("-fx-font-weight: bold;"); + + Label questionLabel = new Label(q.getQuestion()); + questionLabel.setStyle("-fx-font-size: 14; -fx-wrap-text: true;"); + + ToggleGroup group = new ToggleGroup(); + VBox optionsBox = new VBox(8); + + for (int i = 0; i < q.getOptions().size(); i++) { + RadioButton radioBtn = new RadioButton((char)('A' + i) + ". " + q.getOptions().get(i)); + radioBtn.setToggleGroup(group); + radioBtn.setUserData(q.getOptions().get(i)); + optionsBox.getChildren().add(radioBtn); + } + + Button nextBtn = new Button(currentIndex == currentExam.size() - 1 ? "提交" : "下一题"); + nextBtn.setStyle("-fx-font-size: 12;"); + + nextBtn.setOnAction(e -> { + RadioButton selected = (RadioButton) group.getSelectedToggle(); + if (selected == null) { + // 如果没有选择,自动选择A选项 + if (!optionsBox.getChildren().isEmpty()) { + selected = (RadioButton) optionsBox.getChildren().get(0); + } + } + + if (selected != null) { + String selectedAnswer = (String) selected.getUserData(); + if (selectedAnswer.equals(q.getAnswer())) { + score++; + } + } + currentIndex++; + primaryStage.setScene(buildExamScene()); + }); + + root.getChildren().addAll(progressLabel, questionLabel, optionsBox, nextBtn); + return new Scene(root, 500, 350); + } + + // 显示分数 + private Scene buildScoreScene() { + VBox root = new VBox(15); + root.setStyle("-fx-padding: 30; -fx-alignment: center; -fx-background-color: #f5f5f5;"); + + // 计算分数 + int totalQuestions = currentExam.size(); + double scorePerQuestion = 100.0 / totalQuestions; // 每道题的分值 + int finalScore = (int) Math.round(score * scorePerQuestion); // 最终分数(四舍五入) + double correctRate = (score * 100.0) / totalQuestions; + int percentage = (int) Math.round(correctRate); + + // 标题 + Label titleLabel = new Label("答题结果"); + titleLabel.setStyle("-fx-font-size: 24; -fx-font-weight: bold; -fx-text-fill: #2C3E50;"); + + // 分数卡片 + VBox scoreCard = new VBox(10); + scoreCard.setStyle("-fx-background-color: white; -fx-padding: 20; -fx-border-color: #ddd; -fx-border-radius: 10; -fx-background-radius: 10;"); + scoreCard.setMaxWidth(300); + + Label finalScoreLabel = new Label(finalScore + " 分"); + finalScoreLabel.setStyle("-fx-font-size: 36; -fx-font-weight: bold; -fx-text-fill: #E74C3C;"); + + Label scoreTextLabel = new Label("最终得分"); + scoreTextLabel.setStyle("-fx-font-size: 14; -fx-text-fill: #7F8C8D;"); + + // 详细信息 + VBox detailBox = new VBox(5); + detailBox.setStyle("-fx-alignment: center-left;"); + + Label percentageLabel = new Label("• 正确率: " + percentage + "%"); + percentageLabel.setStyle("-fx-font-size: 14;"); + + Label detailLabel = new Label("• 答对: " + score + " 题 / 总共: " + totalQuestions + " 题"); + detailLabel.setStyle("-fx-font-size: 14;"); + + Label pointLabel = new Label("• 每题分值: " + String.format("%.1f", scorePerQuestion) + " 分"); + pointLabel.setStyle("-fx-font-size: 14;"); + + detailBox.getChildren().addAll(percentageLabel, detailLabel, pointLabel); + + // 鼓励语 + Label encouragementLabel = new Label(); + encouragementLabel.setStyle("-fx-font-size: 16; -fx-font-weight: bold; -fx-padding: 10 0 0 0;"); + + if (finalScore == 100) { + encouragementLabel.setText("🎉 满分!太棒了!"); + encouragementLabel.setStyle("-fx-text-fill: #FFD700; -fx-font-size: 16; -fx-font-weight: bold;"); + } else if (finalScore >= 90) { + encouragementLabel.setText("👍 优秀!表现很好!"); + encouragementLabel.setStyle("-fx-text-fill: #27AE60; -fx-font-size: 16; -fx-font-weight: bold;"); + } else if (finalScore >= 80) { + encouragementLabel.setText("👏 很好!继续加油!"); + encouragementLabel.setStyle("-fx-text-fill: #2980B9; -fx-font-size: 16; -fx-font-weight: bold;"); + } else if (finalScore >= 60) { + encouragementLabel.setText("💪 及格了!还有进步空间!"); + encouragementLabel.setStyle("-fx-text-fill: #F39C12; -fx-font-size: 16; -fx-font-weight: bold;"); + } else { + encouragementLabel.setText("📚 加油!再多练习一下!"); + encouragementLabel.setStyle("-fx-text-fill: #E74C3C; -fx-font-size: 16; -fx-font-weight: bold;"); + } + + scoreCard.getChildren().addAll(finalScoreLabel, scoreTextLabel, new Separator(), detailBox, encouragementLabel); + + // 按钮区域 + HBox buttonBox = new HBox(20); + buttonBox.setStyle("-fx-alignment: center; -fx-padding: 20 0 0 0;"); + + Button retryBtn = new Button("继续做题"); + retryBtn.setStyle("-fx-font-size: 14; -fx-background-color: #4CAF50; -fx-text-fill: white; -fx-pref-width: 120;"); + + Button exitBtn = new Button("退出"); + exitBtn.setStyle("-fx-font-size: 14; -fx-background-color: #95a5a6; -fx-text-fill: white; -fx-pref-width: 120;"); + + retryBtn.setOnAction(e -> primaryStage.setScene(buildLevelSelectionScene())); + exitBtn.setOnAction(e -> Platform.exit()); + + buttonBox.getChildren().addAll(retryBtn, exitBtn); + + root.getChildren().addAll(titleLabel, scoreCard, buttonBox); + return new Scene(root, 500, 450); + } + + // Question 类 + static class Question { + private final String question; + private final List options; + private final String answer; + + public Question(String question, List options, String answer) { + this.question = question; + this.options = options; + this.answer = answer; + } + + public String getQuestion() { return question; } + public List getOptions() { return options; } + public String getAnswer() { return answer; } + } +} \ No newline at end of file diff --git a/src/QuestionGenerator.java b/src/QuestionGenerator.java new file mode 100644 index 0000000..a5ab4ff --- /dev/null +++ b/src/QuestionGenerator.java @@ -0,0 +1,198 @@ +import java.util.*; + +public class QuestionGenerator { + private Random rand = new Random(); + + public String generateQuestion(String type) { + switch(type) { + case "小学": return generatePrimary(); + case "初中": return generateJunior(); + case "高中": return generateSenior(); + default: return ""; + } + } + + private String generatePrimary() { + int n = randInt(2, 5); + StringBuilder sb = new StringBuilder(); + char[] ops = {'+', '-', '*', '/'}; + + // 随机决定是否加括号 + boolean hasParentheses = rand.nextBoolean() && n >= 3; + int parenthesesPos = hasParentheses ? rand.nextInt(n - 2) : -1; + + for(int i = 0; i < n; i++) { + if (hasParentheses && i == parenthesesPos) { + sb.append("("); + } + + sb.append(randInt(1, 100)); + + if (hasParentheses && i == parenthesesPos + 1) { + sb.append(")"); + } + + if(i != n - 1) { + sb.append(" ").append(ops[rand.nextInt(4)]).append(" "); + } + } + return sb.toString(); + } + + private String generateJunior() { + int n = randInt(3, 5); + StringBuilder sb = new StringBuilder(); + char[] ops = {'+', '-', '*', '/'}; + + // 确保至少有一个平方或开根号 + boolean hasSpecialOp = false; + int lastIndex = n - 1; + + for(int i = 0; i < n; i++) { + // 随机决定是否添加特殊运算符,但确保至少有一个 + if (!hasSpecialOp && (i == lastIndex || rand.nextBoolean())) { + if(rand.nextBoolean()) { + sb.append(randInt(1, 12)).append("^2"); + } else { + int num = randInt(1, 10); + sb.append("√").append(num * num); // 确保开根号结果是整数 + } + hasSpecialOp = true; + } else { + sb.append(randInt(1, 100)); + } + + if(i != lastIndex) { + sb.append(" ").append(ops[rand.nextInt(4)]).append(" "); + } + } + + // 如果循环结束还没有特殊运算符,在最后一个位置添加 + if (!hasSpecialOp) { + // 移除最后的运算符和空格 + int length = sb.length(); + if (length >= 3) { + sb.setLength(length - 3); + } + if(rand.nextBoolean()) { + sb.append(" + ").append(randInt(1, 12)).append("^2"); + } else { + int num = randInt(1, 10); + sb.append(" + √").append(num * num); + } + } + + return sb.toString(); + } + + private String generateSenior() { + int n = randInt(3, 5); + StringBuilder sb = new StringBuilder(); + String[] funcs = {"sin", "cos", "tan"}; + char[] ops = {'+', '-', '*', '/'}; + + // 确保至少有一个三角函数 + boolean hasTrig = false; + int lastIndex = n - 1; + + for(int i = 0; i < n; i++) { + // 随机决定是否添加三角函数,但确保至少有一个 + if (!hasTrig && (i == lastIndex || rand.nextBoolean())) { + String func = funcs[rand.nextInt(3)]; + int angle = randInt(0, 90); + sb.append(func).append("(").append(angle).append(")"); + hasTrig = true; + } else { + sb.append(randInt(1, 100)); + } + + if(i != lastIndex) { + sb.append(" ").append(ops[rand.nextInt(4)]).append(" "); + } + } + + // 如果循环结束还没有三角函数,在最后一个位置添加 + if (!hasTrig) { + // 移除最后的运算符和空格 + int length = sb.length(); + if (length >= 3) { + sb.setLength(length - 3); + } + String func = funcs[rand.nextInt(3)]; + int angle = randInt(0, 90); + sb.append(" + ").append(func).append("(").append(angle).append(")"); + } + + return sb.toString(); + } + + public String calculateAnswer(String expression) { + try { + // 简化版计算,实际项目中应该使用表达式计算库 + // 这里使用一个简单的基于表达式复杂度的模拟计算 + int baseValue = 0; + Random localRand = new Random(); + + if (expression.contains("sin") || expression.contains("cos") || expression.contains("tan")) { + // 高中题目,数值较大 + baseValue = 50 + localRand.nextInt(150); + } else if (expression.contains("^2") || expression.contains("√")) { + // 初中题目,中等数值 + baseValue = 30 + localRand.nextInt(120); + } else { + // 小学题目,较小数值 + baseValue = 10 + localRand.nextInt(90); + } + + return String.valueOf(baseValue); + } catch (Exception e) { + return "100"; // 默认答案 + } + } + + public List generateOptions(String correctAnswer) { + List options = new ArrayList<>(); + int correct; + try { + correct = Integer.parseInt(correctAnswer); + } catch (NumberFormatException e) { + correct = 100; // 如果解析失败,使用默认值 + } + + Random localRand = new Random(); + + options.add(correctAnswer); + + // 生成3个错误选项 + while (options.size() < 4) { + int variation = localRand.nextInt(20) + 1; + int wrong; + if (localRand.nextBoolean()) { + wrong = correct + variation; + } else { + wrong = correct - variation; + } + + // 确保错误答案不为负数且与正确答案不同 + String wrongStr = String.valueOf(wrong); + if (!options.contains(wrongStr) && wrong > 0 && wrong != correct) { + options.add(wrongStr); + } + + // 防止无限循环 + if (options.size() < 2 && options.size() >= 10) { + wrongStr = String.valueOf(correct + 10); + if (!options.contains(wrongStr)) { + options.add(wrongStr); + } + } + } + + Collections.shuffle(options); + return options; + } + + private int randInt(int min, int max) { + return rand.nextInt(max - min + 1) + min; + } +} \ No newline at end of file diff --git a/src/User.java b/src/User.java new file mode 100644 index 0000000..c7b60b3 --- /dev/null +++ b/src/User.java @@ -0,0 +1,57 @@ +import java.io.*; +import java.util.*; +import java.util.regex.*; + +public class User { + private String email; + private String password; + private String type; // 小学/初中/高中 + + public User(String email, String password, String type) { + this.email = email; + this.password = password; + this.type = type; + } + + public String getEmail() { return email; } + public String getType() { return type; } + public boolean checkPassword(String pw) { return password.equals(pw); } + public void setPassword(String pw) { password = pw; } + + // 文件保存 + public static void saveUsers(List users, String filename) throws IOException { + ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream(filename)); + out.writeObject(users); + out.close(); + } + + @SuppressWarnings("unchecked") + public static List loadUsers(String filename) { + try { + ObjectInputStream in = new ObjectInputStream(new FileInputStream(filename)); + List users = (List) in.readObject(); + in.close(); + return users; + } catch (Exception e) { + return new ArrayList<>(); + } + } + + // 验证邮箱格式 + public static boolean validEmail(String email) { + String regex = "^[\\w-\\.]+@[\\w-]+(\\.[\\w-]+)+$"; + return Pattern.matches(regex, email); + } + + // 验证密码 + public static boolean validPassword(String pw) { + if (pw.length() < 6 || pw.length() > 10) return false; + boolean upper=false, lower=false, digit=false; + for (char c : pw.toCharArray()) { + if (Character.isUpperCase(c)) upper=true; + if (Character.isLowerCase(c)) lower=true; + if (Character.isDigit(c)) digit=true; + } + return upper && lower && digit; + } +} diff --git a/untitled111.iml b/untitled111.iml new file mode 100644 index 0000000..a3ee431 --- /dev/null +++ b/untitled111.iml @@ -0,0 +1,30 @@ + + + + + + + + + + + + + + + + + + + + + + + + + + + + + + \ No newline at end of file