我使用扫描器方法nextInt()和nextLine()读取输入。

它是这样的:

System.out.println("Enter numerical value");    
int option;
option = input.nextInt(); // Read numerical value from input
System.out.println("Enter 1st string"); 
String string1 = input.nextLine(); // Read 1st string (this is skipped)
System.out.println("Enter 2nd string");
String string2 = input.nextLine(); // Read 2nd string (this appears right after reading numerical value)

问题是在输入数值之后,第一个input.nextLine()被跳过,第二个input.nextLine()被执行,因此我的输出看起来像这样:

Enter numerical value
3   // This is my input
Enter 1st string    // The program is supposed to stop here and wait for my input, but is skipped
Enter 2nd string    // ...and this line is executed and waits for my input

我测试了我的应用程序,看起来问题在于使用input.nextInt()。如果我删除它,那么string1 = input.nextLine()和string2 = input.nextLine()都按照我想要的方式执行。


问题出在input.nextInt()方法;它只读取int值。因此,当您继续使用input.nextLine()读取时,您会收到“\n”输入键。所以要跳过这个,你必须添加input.nextLine()。

试试这样吧:

System.out.print("Insert a number: ");
int number = input.nextInt();
input.nextLine(); // This line you have to add (It consumes the \n character)
System.out.print("Text1: ");
String text1 = input.nextLine();
System.out.print("Text2: ");
String text2 = input.nextLine();

这是因为当您输入一个数字然后按enter时,input.nextInt()只使用该数字,而不是“行结束”。当input. nextline()执行时,它使用仍然在第一个输入缓冲区中的“行尾”。

相反,在input.nextInt()之后立即使用input.nextLine()


那是因为扫描仪。nextInt方法不读取通过按“Enter”创建的输入中的换行符,因此调用Scanner。nextLine在读取换行符后返回。

当您使用Scanner时,您将遇到类似的行为。在Scanner.next()或任何Scanner之后的nextLine。nextFoo方法(除了nextLine本身)。

处理:

Either put a Scanner.nextLine call after each Scanner.nextInt or Scanner.nextFoo to consume rest of that line including newline int option = input.nextInt(); input.nextLine(); // Consume newline left-over String str1 = input.nextLine(); Or, even better, read the input through Scanner.nextLine and convert your input to the proper format you need. For example, you may convert to an integer using Integer.parseInt(String) method. int option = 0; try { option = Integer.parseInt(input.nextLine()); } catch (NumberFormatException e) { e.printStackTrace(); } String str1 = input.nextLine();


这样做是因为input.nextInt();不捕获换行符。你可以像其他人一样通过添加input.nextLine();在下面。 或者你也可以用c#风格,把nextLine解析成一个整数,如下所示:

int number = Integer.parseInt(input.nextLine()); 

这样做效果很好,而且节省了一行代码。


关于java.util.Scanner的这个问题似乎有很多问题。我认为一个更可读/惯用的解决方案是调用scanner.skip("[\r\n]+")在调用nextInt()后删除任何换行符。

编辑:正如下面提到的@PatrickParker,如果用户在数字后输入任何空白,这将导致无限循环。关于更好的skip模式,请参阅他们的回答:https://stackoverflow.com/a/42471816/143585


使用input.next()而不是input.nextLine(),这应该可以解决问题。

修改代码:

public static Scanner input = new Scanner(System.in);

public static void main(String[] args)
{
    System.out.print("Insert a number: ");
    int number = input.nextInt();
    System.out.print("Text1: ");
    String text1 = input.next();
    System.out.print("Text2: ");
    String text2 = input.next();
}

为什么不使用一个新的扫描器每次读取?像下面。用这种方法你就不会直面你的问题。

int i = new Scanner(System.in).nextInt();

博士TL; 当(a)它是第一个读取指令,(b)之前的读取指令也是nextLine()时,调用nextLine()是安全的。

如果你不确定以上任何一个是正确的,你可以在调用scanner.nextLine()之前使用scanner.skip("\\R?"),因为调用next() nextInt()将留下潜在的行分隔符-由返回键创建,这将影响nextLine()的结果。.skip("\\R?")将让我们使用这个不必要的行分隔符。

Skip使用正则表达式where

\R表示行分隔符 ? 将使\R是可选的-这将防止跳过方法: 等待匹配序列 在系统等仍然开放的数据源到达终点时。在,输入流从套接字等。 抛出java.util.NoSuchElementException 终止/关闭的数据来源, 或者当现有数据与我们想要跳过的内容不匹配时

你需要知道的事情:

表示几行的文本在行之间也包含不可打印的字符(我们称之为行分隔符) 回车(CR -在字符串中以“\r”表示) 换行(LF - in字符串字面值表示为“\n”) 当您从控制台读取数据时,它允许用户输入他的响应,当他完成时,他需要以某种方式确认这一事实。为此,用户需要按下键盘上的“enter”/“return”键。

重要的是,除了确保将用户数据放置到标准输入(由System。(由Scanner读取)还会在后面发送与操作系统相关的行分隔符(如Windows \r\n)。

因此,当您向用户询问年龄等值时,用户输入42并按enter,标准输入将包含“42\r\n”。

问题

Scanner#nextInt(以及其他Scanner#nextType方法)不允许Scanner使用这些行分隔符。它将从系统中读取它们。in(否则Scanner如何知道用户没有更多的数字表示年龄值而不是面对空白?),这将从标准输入中删除它们,但它也将在内部缓存这些行分隔符。我们需要记住的是,所有的Scanner方法总是从缓存的文本开始扫描。

现在Scanner#nextLine()只是收集并返回所有字符,直到找到行分隔符(或流的结束)。但是由于从控制台读取数字后的行分隔符立即在Scanner的缓存中找到,因此它返回空String,这意味着Scanner无法在这些行分隔符(或流的结束)之前找到任何字符。 顺便说一句,nextLine也会使用这些行分隔符。

解决方案

当你想要求number,然后求整行同时避免作为nextLine结果的空字符串

consume line separator left by nextInt from Scanners cache by calling nextLine, or IMO more readable way would be by calling skip("\\R") or skip("\r\n|\r|\n") to let Scanner skip part matched by line separator (more info about \R: https://stackoverflow.com/a/31060125) don't use nextInt (nor next, or any nextTYPE methods) at all. Instead read entire data line-by-line using nextLine and parse numbers from each line (assuming one line contains only one number) to proper type like int via Integer.parseInt.


顺便说一句:扫描器#nextType方法可以跳过分隔符(默认情况下所有空白,如制表符,行分隔符),包括扫描器缓存的那些,直到它们找到下一个非分隔符值(令牌)。感谢它输入像“42\r\n\r\n321\r\n\r\n\r\nfoobar”代码

int num1 = sc.nextInt();
int num2 = sc.nextInt();
String name = sc.next();

将能够正确地分配num1=42 num2=321 name=foobar。


为了避免这个问题,请使用nextLine();紧接在nextInt()之后;因为它有助于清除缓冲区。当你按ENTER时,nextInt();不会捕获新行,因此稍后将跳过Scanner代码。

Scanner scanner =  new Scanner(System.in);
int option = scanner.nextInt();
scanner.nextLine(); //clearing the buffer

如果你想快速扫描输入而不混淆到扫描器类nextLine()方法,请使用自定义输入扫描器。

代码:

class ScanReader {
/**
* @author Nikunj Khokhar
*/
    private byte[] buf = new byte[4 * 1024];
    private int index;
    private BufferedInputStream in;
    private int total;

    public ScanReader(InputStream inputStream) {
        in = new BufferedInputStream(inputStream);
    }

    private int scan() throws IOException {
        if (index >= total) {
            index = 0;
            total = in.read(buf);
            if (total <= 0) return -1;
        }
        return buf[index++];
    }
    public char scanChar(){
        int c=scan();
        while (isWhiteSpace(c))c=scan();
        return (char)c;
    }


    public int scanInt() throws IOException {
        int integer = 0;
        int n = scan();
        while (isWhiteSpace(n)) n = scan();
        int neg = 1;
        if (n == '-') {
            neg = -1;
            n = scan();
        }
        while (!isWhiteSpace(n)) {
            if (n >= '0' && n <= '9') {
                integer *= 10;
                integer += n - '0';
                n = scan();
            }
        }
        return neg * integer;
    }

    public String scanString() throws IOException {
        int c = scan();
        while (isWhiteSpace(c)) c = scan();
        StringBuilder res = new StringBuilder();
        do {
            res.appendCodePoint(c);
            c = scan();
        } while (!isWhiteSpace(c));
        return res.toString();
    }

    private boolean isWhiteSpace(int n) {
        if (n == ' ' || n == '\n' || n == '\r' || n == '\t' || n == -1) return true;
        else return false;
    }

    public long scanLong() throws IOException {
        long integer = 0;
        int n = scan();
        while (isWhiteSpace(n)) n = scan();
        int neg = 1;
        if (n == '-') {
            neg = -1;
            n = scan();
        }
        while (!isWhiteSpace(n)) {
            if (n >= '0' && n <= '9') {
                integer *= 10;
                integer += n - '0';
                n = scan();
            }
        }
        return neg * integer;
    }

    public void scanLong(long[] A) throws IOException {
        for (int i = 0; i < A.length; i++) A[i] = scanLong();
    }

    public void scanInt(int[] A) throws IOException {
        for (int i = 0; i < A.length; i++) A[i] = scanInt();
    }

    public double scanDouble() throws IOException {
        int c = scan();
        while (isWhiteSpace(c)) c = scan();
        int sgn = 1;
        if (c == '-') {
            sgn = -1;
            c = scan();
        }
        double res = 0;
        while (!isWhiteSpace(c) && c != '.') {
            if (c == 'e' || c == 'E') {
                return res * Math.pow(10, scanInt());
            }
            res *= 10;
            res += c - '0';
            c = scan();
        }
        if (c == '.') {
            c = scan();
            double m = 1;
            while (!isWhiteSpace(c)) {
                if (c == 'e' || c == 'E') {
                    return res * Math.pow(10, scanInt());
                }
                m /= 10;
                res += (c - '0') * m;
                c = scan();
            }
        }
        return res * sgn;
    }

}

优点:

扫描输入比BufferReader更快 降低时间复杂度 为每个下一个输入刷新缓冲区

方法:

scanChar() -扫描单个字符 scanInt() - scan整数值 scanLong() -扫描长值 scanString() -扫描字符串值 scanouble () - scan双值 scanInt(int[] array) -扫描完整的数组(整数) scanLong(long[] array) -扫描完整的array (long)

用法:

复制java代码下面的给定代码。 初始化给定类的对象

ScanReader sc = new ScanReader(System.in); 3.导入必要的类:

进口java.io.BufferedInputStream; 进口java.io.IOException; 进口java.io.InputStream; 4. 从主方法中抛出IOException来处理Exception 5. 使用提供的方法。 6. 享受

例子:

import java.io.BufferedInputStream;
import java.io.IOException;
import java.io.InputStream;
class Main{
    public static void main(String... as) throws IOException{
        ScanReader sc = new ScanReader(System.in);
        int a=sc.scanInt();
        System.out.println(a);
    }
}
class ScanReader....

如果你想同时读取字符串和int,一个解决方案是使用两个扫描仪:

Scanner stringScanner = new Scanner(System.in);
Scanner intScanner = new Scanner(System.in);

intScanner.nextInt();
String s = stringScanner.nextLine(); // unaffected by previous nextInt()
System.out.println(s);

intScanner.close();
stringScanner.close();

public static void main(String[] args) {
        Scanner scan = new Scanner(System.in);
        int i = scan.nextInt();
        scan.nextLine();
        double d = scan.nextDouble();
        scan.nextLine();
        String s = scan.nextLine();

        System.out.println("String: " + s);
        System.out.println("Double: " + d);
        System.out.println("Int: " + i);
    }

与解析输入相比,sc.nextLine()更好。 因为就性能而言,它会很好。


我想我来派对已经很晚了。

如前所述,在获取int值后调用input.nextLine()将解决您的问题。您的代码不能工作的原因是因为没有其他东西可以从您的输入(您输入int的地方)存储到string1。我将对整个主题进行更多的阐述。

将nextLine()视为Scanner类中nextFoo()方法中的一个特例。让我们举一个简单的例子。假设我们有如下两行代码:

int firstNumber = input.nextInt();
int secondNumber = input.nextInt();

如果我们输入下面的值(作为单行输入)

54 234

firstNumber和secondNumber变量的值分别变为54和234。这种工作方式的原因是因为当nextInt()方法接收值时,不会自动生成新的换行(即\n)。它只是接受“下一个int”并继续前进。除了nextLine(),其余的nextFoo()方法也是如此。

nextLine()在取值后立即生成新的换行;这就是@RohitJain说新的换行被“消耗”的意思。

最后,next()方法只接受最近的String而不生成新行;这使得它成为在同一行中获取单独字符串的优先方法。

我希望这能有所帮助。编码快乐!


使用两个扫描仪对象而不是一个

Scanner input = new Scanner(System.in);
System.out.println("Enter numerical value");    
int option;
Scanner input2 = new Scanner(System.in);
option = input2.nextInt();

如果我期望一个非空的输入

避免: -如果以下输入被未检查的scan.nextLine()作为解决方法吃掉,则数据丢失 因为scan.nextLine()被scan.next()取代而导致的数据丢失(输入:"yippie ya yeah") -使用Scanner方法解析输入时抛出的异常(先读取,然后解析)

public static Function<Scanner,String> scanLine = (scan -> {
    String s = scan.nextLine();
    return( s.length() == 0 ? scan.nextLine() : s );
  });

用于上面的例子:

System.out.println("Enter numerical value");    
int option = input.nextInt(); // read numerical value from input

System.out.println("Enter 1st string"); 
String string1 = scanLine.apply( input ); // read 1st string
System.out.println("Enter 2nd string");
String string2 = scanLine.apply( input ); // read 2nd string

因为nextXXX()方法不读取换行符,除了nextLine()。我们可以在读取任何非字符串值(在这种情况下是int)后跳过换行符,使用scanner.skip()如下所示:

Scanner sc = new Scanner(System.in);
int x = sc.nextInt();
sc.skip("(\r\n|[\n\r\u2028\u2029\u0085])?");
System.out.println(x);
double y = sc.nextDouble();
sc.skip("(\r\n|[\n\r\u2028\u2029\u0085])?");
System.out.println(y);
char z = sc.next().charAt(0);
sc.skip("(\r\n|[\n\r\u2028\u2029\u0085])?");
System.out.println(z);
String hello = sc.nextLine();
System.out.println(hello);
float tt = sc.nextFloat();
sc.skip("(\r\n|[\n\r\u2028\u2029\u0085])?");
System.out.println(tt);

在我的一个用例中,我有这样的场景:读取一个字符串值之前有几个整数值。我必须使用“for / while循环”来读取值。以上的建议在这个案例中都不起作用。

使用input.next()而不是input.nextLine()修复了这个问题。希望这对那些遇到类似情况的人有所帮助。


使用此代码,它将解决您的问题。

System.out.println("Enter numerical value");    
int option;
option = input.nextInt(); // Read numerical value from input
input.nextLine();
System.out.println("Enter 1st string"); 
String string1 = input.nextLine(); // Read 1st string (this is skipped)
System.out.println("Enter 2nd string");
String string2 = input.nextLine(); // Read 2nd string (this appears right after reading numerical value)

问题出在input.nextInt()方法上——它只读取int值。因此,当您继续使用input.nextLine()读取时,您会收到“\n”输入键。所以要跳过这个,你必须添加input.nextLine()。希望这一点现在应该清楚了。

试着这样做:

System.out.print("Insert a number: ");
int number = input.nextInt();
input.nextLine(); // This line you have to add (It consumes the \n character)
System.out.print("Text1: ");
String text1 = input.nextLine();
System.out.print("Text2: ");
String text2 = input.nextLine();

要解决这个问题,只需创建scan. nextline(),其中scan是Scanner对象的一个实例。例如,我使用一个简单的HackerRank问题来解释。

package com.company;
import java.util.Scanner;

public class hackerrank {
public static void main(String[] args) {
    Scanner scan = new Scanner(System.in);
    int i = scan.nextInt();
    double d = scan.nextDouble();
    scan.nextLine(); // This line shall stop the skipping the nextLine() 
    String s = scan.nextLine();
    scan.close();



    // Write your code here.

    System.out.println("String: " + s);
    System.out.println("Double: " + d);
    System.out.println("Int: " + i);
}

}


nextLine()将直接将enter读取为空行,而不等待文本。

简单的解决方案是添加一个额外的扫描器来消耗空行:

System.out.println("Enter numerical value");    
int option;
option = input.nextInt(); // Read numerical value from input
input.nextLine();
System.out.println("Enter 1st string"); 
String string1 = input.nextLine(); // Read 1st string (this is skipped)
System.out.println("Enter 2nd string");
String string2 = input.nextLine(); // Read 2nd string (this appears right after reading numerical value)

对于java初学者来说,这是一个非常基本的问题。当我开始学习java(自学)时,我也遇到过同样的问题。 实际上,当我们获取一个整数dataType的输入时,它只读取整数值,并留下newLine(\n)字符和这一行(即。留下新的行整数动态输入)会在我们尝试新的输入时产生问题。 如。比如,如果我们取一个整数输入然后尝试取一个字符串输入。

value1=sc.nextInt();
value2=sc.nextLine();

value2将自动读取换行符,而不接受用户输入。

解决方案: 我们只需要在获取下一个用户输入之前添加一行代码。

sc.nextLine();

or

value1=sc.nextInt();
sc.nextLine();
value2=sc.nextLine();

注意:不要忘记关闭Scanner,防止内存泄漏;


 Scanner scan = new Scanner(System.in);
    int i = scan.nextInt();
    scan.nextLine();//to Ignore the rest of the line after  (integer input)nextInt()
    double d=scan.nextDouble();
    scan.nextLine();
    String s=scan.nextLine();
    scan.close();
    System.out.println("String: " + s);
    System.out.println("Double: " + d);
    System.out.println("Int: " + i);