Java之Scanner.nextLine,读取回车的问题及解决

问题描述

我们在使用java读取键盘输入时,如果先读取一个int变量,再读取下一行的字符串时,会发现程序运行结果与预期不符,程序并没有读取到下一行的字符串。

即发生在如下情形:

Scanner sc=new Scanner(System.in);
int n=sc.nextInt();
String s=sc.nextLine();

问题分析

从代码分析我们的需求是先输入一个int变量,再在下一行输入一个字符串,也就是说我们使用键盘输入的时候,其实输入了三部分:

3

words

分别是:数字3,换行符,字符串words。

如何处理多出来的换行符

从刚才的分析可以看到,问题的原因是nextLine()方法没有读取到我们的字符串,而是读入了我们的Enter换行符。

方法一

既然多出了一个换行符,那我们可以在读取字符串之前将换行符读掉:

Scanner sc=new Scanner(System.in);
int n=sc.nextInt();
sc.nextLine();//将换行符读掉
String s=sc.nextLine();

方法二

重新创建Scanner对象:

Scanner sc1=new Scanner(System.in);
int n=sc1.nextInt();
Scanner sc2=new Scanner(System.in);
String s=sc2.nextLine();

这样在读取完n之后,换行符会被丢弃,下一行被视作sc2开始读入的位置。

实战

下面是tengxun公司2020年4月26日暑期软件相关岗位实习笔试的第一题:

数据结构基础之一一队列

队列有五种基本操作,插入队尾、取出队首、删除队首、队列大小、清空队列。

现在让你模拟一个队列的操作,具体格式参考输入。

输入描述:

第一行输入一个整数T,表示接下来有T组测试数据。

对于每组测试数据:

第一行输入一个整数Q,表示有Q次操作。

接下来Q行,每行输入一种队列操作方式,具体格式如下:

初始状态下队列为空。

插入队尾: PUSH X

取出队首: TOP//仅仅是看一下队首元素,不要把队首元素删除

删除队首: POP

队列大小: SIZE

清空队列: CLEAR

1<T<100

1<Q,X≤1000

保证操作为以上5种的任意一种。

输出描述:

对于每组测试数据:

如果操作为“取出队首”,输出队首元素,如果无法取出,输出“-1”

如果操作为“删除队首”,如果无法删除,输出“-1”

如果操作为“队列大小”,输出队列大小

其他操作无需输出

示例输入

2

7

PUSH 1

PUSH 2

TOP

POP

TOP

POP

POP

5

PUSH 1

PUSH 2

SIZE

POP

SIZE

结果:

1

2

-1

2

1

问题解答

这道题并不难,使用LinkedList就可以实现上述操作,当然,Queue的底层实现本来也就是LinkedList。

之所以列举这道题是因为这道题会出现nextLine()读取失败的问题,因为我们首先要读取整形变量T,之后再往后逐行读取,如果不处理换行符的问题,将导致程序运行出现异常。

我的AC答案:

import java.util.LinkedList;
import java.util.List;
import java.util.Scanner;

public class Main {
    public static void main(String[] args) {
        Scanner sc =new Scanner(System.in);
        int t=sc.nextInt();
        while(t>0){
            int q=sc.nextInt();
            List<Integer> n=new LinkedList<>();
            for(int i=0;i<=q;i++){//注意!i<=q,这里的‘='就是因为q和第一行输入之间的换行符
                String s=sc.nextLine();
                if(s.equals("")){//在这道题中,当我读取到换行符时,直接跳过,不做任何操作
                    continue;
                }
                if(s.startsWith("PUSH")){
                    n.add(Integer.parseInt(s.split(" ")[1]));
                }
                else if(s.equals("TOP")){
                    if(n.size()==0){
                        System.out.println(-1);
                    }
                    else
                        System.out.println(n.get(0));
                }
                else if(s.equals("POP")){
                    if(n.isEmpty()){
                        System.out.println(-1);
                    }
                    else
                        n.remove(0);
                }
                else if(s.equals("SIZE")){
                    System.out.println(n.size());
                }
                else if(s.equals("CLEAR")){
                    n.clear();
                }
            }
            t--;
        }
    }
}

在上面的代码中,我首先使用了

for(int i=0;i<=q;i++)

注意到了q与接下来的输入之间的换行符。

之后,当遇到换行符时:

String s=sc.nextLine();
if(s.equals(""))
    continue;

来解决该问题。

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持。

原文地址:https://blog.csdn.net/weixin_44388094/article/details/105825740