Scanner.nextLine() 会跳过输入是因为 nextInt() 不消费换行符,导致 nextLine() 立即读到空行;应在其后加 nextLine() 清缓存,或统一用 nextLine() 配合 parseXXX() 转换,避免混用方法。

Scanner.nextLine() 为什么会跳过输入
这是 Java 初学者最常踩的坑:用 nextInt() 或 nextDouble() 读数字后,紧接着调用 nextLine(),结果后者直接返回空字符串。根本原因是 nextInt() 不消费输入末尾的换行符(\n),而 nextLine() 会读取并返回从当前位置到下一个 \n 的所有字符——此时光标还在上一行末尾,所以立刻碰到换行,返回空串。
实操建议:
立即学习“Java免费学习笔记(深入)”;
- 在
nextInt()等之后手动加一句scanner.nextLine()清掉残留换行符 - 统一用
nextLine()读所有输入,再用Integer.parseInt()、Double.parseDouble()转换——更安全,也避免类型不匹配异常 - 不要混用
next()(只读空白前单词)和nextLine(),除非你明确知道当前缓冲区状态
如何安全读取含空格的字符串(比如姓名、地址)
next() 遇到空格就停,根本读不全;nextLine() 才是正确选择。但它对“输入流已关闭”或“用户提前终止”很敏感,容易抛 NoSuchElementException。
实操建议:
立即学习“Java免费学习笔记(深入)”;
- 始终用
hasNextLine()做前置检查,尤其在循环中 - 如果业务允许,用
nextLine().trim()去除首尾空格,避免空输入被当成有效数据 - 遇到必须解析结构化文本(如“张三,25,北京”),别硬拆
nextLine(),改用String.split(",")+ 异常捕获
Scanner 关闭后还能不能再次使用
不能。scanner.close() 会关闭其底层的 System.in 流——这个流是 JVM 全局的,一旦关了,后续任何新 Scanner 实例都会在第一次调用 nextXXX() 时抛 IllegalStateException: Scanner closed。
实操建议:
立即学习“Java免费学习笔记(深入)”;
- 一个程序里只创建一个
Scanner实例,并在整个生命周期内复用 - 绝对不要在方法末尾写
scanner.close(),除非你 100% 确认这是最后一次使用System.in - 如果真要释放资源(比如单元测试中),改用
try-with-resources包裹,但仅限于明确“本次输入彻底结束”的场景
public class InputDemo {
private static final Scanner scanner = new Scanner(System.in); // 单例复用
public static void main(String[] args) {
System.out.print("请输入年龄:");
if (scanner.hasNextInt()) {
int age = scanner.nextInt();
scanner.nextLine(); // 消费换行符
System.out.print("请输入姓名:");
String name = scanner.nextLine().trim();
System.out.println("姓名:" + name + ",年龄:" + age);
} else {
System.out.println("输入不是整数");
scanner.next(); // 消费非法 token,避免死循环
}
// 不 close()
}
}
控制台输入处理真正难的不是语法,而是状态管理:换行符残留、流关闭时机、异常输入的恢复。这些细节不写进日志,也不报错明显,但会让程序在特定输入下静默失败。










