Java初学者项目实战:创建学生信息管理系统

学生信息管理系统需用ArrayList保顺序、HashMap按学号索引实现高效查询;输入统一用nextLine()再转换类型并try-catch;修改学生须操作原对象引用;边界情况如空集合、重复学号等必须处理。

学生信息管理系统不是练手玩具,而是检验你能否把零散知识点串成可用逻辑的关键节点。它不难,但容易在“存哪”“怎么查”“改完不生效”上卡住半天。

用 ArrayList 还是 HashMap 存学生?看你要不要按学号快速查找

初学者常默认用 ArrayList,写起来顺,增删也直观。但一旦要根据学号查学生,就得遍历——for (Student s : list) 一跑就是 O(n)。如果只是课堂演示、数据不到 100 条,没问题;但若加个“按学号查详情”按钮,用户点一下等半秒,体验就垮了。

更合理的做法是双存:用 ArrayList 保顺序(比如按录入时间展示列表),再用 HashMap 做索引,键为学号("2025001")。插入时两处都加,删除时两处都删。别省这十几行代码,后面所有查询操作会感谢你。

  • 学号作 key 时务必校验非空、格式合法,否则 map.get(null) 直接抛 NullPointerException
  • 避免用 new Student() 对象作 map 的 key——没重写 equals()hashCode() 就永远找不到
  • 如果后期要支持模糊搜索(如“张*”),ArrayList 配合 stream().filter() 更灵活

Scanner.nextLine() 吃掉回车?不是 bug,是换行符残留

输入姓名后敲回车,接着输年龄,结果程序跳过年龄直接卡住——这是 Scanner 最经典的坑:nextInt()nextDouble() 不读取结尾的换行符,下次调 nextLine() 就立刻拿到一个空字符串。

解决方法不是换库,而是统一用 nextLine() 读所有输入,再手动转类型:

System.out.print("请输入年龄:");
String ageStr = scanner.nextLine();
int age = Integer.parseInt(ageStr); // 加 try-catch 处理 NumberFormatException

这样逻辑清晰,也不用记哪次该补 scanner.nextLine()

  • 所有用户输入都走 nextLine(),哪怕你只想要一个 Y/N
  • Integer.parseInt()Double.parseDouble() 必须包在 try-catch 里,否则输个“abc”程序就崩
  • 别用 scanner.hasNextInt() 判断再读——它不会消费输入流,下一次读还是那个非法字符串

修改学生信息后界面上没变?对象引用没搞清

你从 ArrayListget(2) 出来一个 Student 对象,改了它的 name 字段,但刷新列表时发现名字还是旧的——大概率是因为你没改原对象,而是 new 了一个新对象赋给了局部变量。

Java 里集合存的是引用。想改真实数据,必须调用原对象的方法,或直接改其 public 字段(不推荐):

Student target = students.get(index);
target.setName("李四"); // ✅ 正确:修改堆中同一对象
// target = new Student(...); // ❌ 错误:只是让局部变量指向新对象,原集合不变

如果你用了 Stream 查找:students.stream().filter(...).findFirst().orElse(null),拿到的也是原对象引用,放心改。

  • 避免在循环里用 list.get(i).setName(...) 同时又 list.remove(i)——索引会错位,用迭代器或倒序遍历
  • 如果学生类字段全用 private + getter/setter,记得检查 setter 里有没有防御性拷贝(比如传入 Date 时 new 一个副本),否则外部改了日期,内部也跟着变

系统跑起来之后,最容易被忽略的是边界:空集合时的提示语、重复学号的拦截逻辑、删除最后一个学生后界面是否清空——这些不是锦上添花,而是用户第一眼看到的“系统稳不稳”的全部依据。