5

我这里有两个代码块。一个扫描仪正确地等待用户输入,而另一个扫描仪直接吹过它并调用nextInt()它返回一个NoSuchElementException. 这是有效的块:

public void startGame() {
   out.println("Player1: 1 for dumb player, 2 for smart player, 3 for human player.");
   Scanner scan = new Scanner(System.in);
   p = scan.nextInt();

   if (p == 1)
        p1 = new DumbPlayer("ONE");
   if (p == 2)
        p1 = new SmartPlayer("ONE");
   else 
       p1 = new HumanPlayer("ONE");

   out.println("Player2: 1 for dumb player, 2 for smart player, 3 for human player.");
   p = scan.nextInt();

   if (p == 1)
        p2 = new DumbPlayer("TWO");
   if (p == 2)
        p2 = new SmartPlayer("TWO");
   else 
        p2 = new HumanPlayer("TWO");

   scan.close();

这是没有的块:

public int findBestMove(Set<Integer> moves, Board b) {

    Set<Integer> set = new HashSet<Integer>();

    out.println("Player " +name+ ", select a column from 1-7: ");
    Scanner scan = new Scanner(System.in);  <--here it should wait for input, but does not!
    int move = scan.nextInt();   <-- NoSuchElementException
    scan.close();

    for (int x = 1; x <= 7; x++) {
        set.add(move);
        move += 7;
    }
    ....etc

这两个都是独立的类,并且是从另一个类的主方法调用的。基本上main()调用startGame(),然后调用findBestMove()一些 Player 类的方法......这是非工作代码所在的位置。程序中是否存在不适合接受输入的情况?我的印象是,只要我想要用户输入,我就可以使用这种方法。谢谢!

4

4 回答 4

6

nextInt()不会丢弃\n在流中,因此下一次调用什么也找不到。您必须使用Scanner.skip("\n")or手动跳过它Scanner.nextLine()

编辑在弄清楚“跳过readInt()”意味着代码抛出异常之后。在@Andreas 的帮助下,这是一个替代解决方案。在 java6 中添加了一个新的类Console来为 stdin 提供更好的接口。它返回一个不关心你关闭它的阅读器。所以下面的代码片段工作得很好:

    Scanner fi = new Scanner(System.console().reader());
    System.out.println(fi.nextInt());
    fi.close();

    fi = new Scanner(System.console().reader());
    System.out.println(fi.nextInt());
    fi.close();
于 2012-11-14T07:53:37.047 回答
5

根据java.util.Scanner javadocScanner.close()如果此流实现Closeable接口,则关闭关联的流。java.lang.System.in是一个InputStream,它实现了Closeable接口。因此,在调用与关联的Scanner.close()a之后,流将关闭并且不再可用。ScannerSystem.inSystem.in

以下 SSCCE 对我有用。我从问题中删除了一些与实际问题无关的代码。请注意,使用这种方法,虽然它有效,但 Eclipse 会给我警告"Resource leak: 'scan' is never closed",因此更好的解决方案是仅使用一个 Scanner 实例。

package com.example;

import java.util.Scanner;
import java.util.Set;
import static java.lang.System.out;

public class ScannerTest {
    int p = 0;
    String name = "Test";

    public void startGame() {
        out.println("Player1: 1 for dumb player, 2 for smart player, 3 for human player.");
        Scanner scan = new Scanner(System.in);
        p = scan.nextInt();

        out.println("Result1: " + p);

        out.println("Player2: 1 for dumb player, 2 for smart player, 3 for human player.");
        p = scan.nextInt();

        out.println("Result2: " + p);

        // scan.close();   // Do not close the Scanner to leave System.in open
    }

    public int findBestMove(Set<Integer> moves, Object /*Board*/ b) {
        out.println("Player " +name+ ", select a column from 1-7: ");
        Scanner scan = new Scanner(System.in);
        int move = scan.nextInt();
        // scan.close();   // Do not close the Scanner to leave System.in open

        out.println("Move: " + move);

        return 0;
    }

    public void run() {
        startGame();
        findBestMove(null, null);
    }

    public static void main(String[] args) {
        ScannerTest st = new ScannerTest();
        st.run();
    }
}
于 2012-11-14T09:00:02.257 回答
0

尝试使用scan.reset();而不是scan.close(). scan.close()正如 Denis Tulskiy 所说,将抛出 NoSuchElementException。

于 2012-11-14T08:35:30.083 回答
0

我只花了大约 15 分钟来解决这个问题,我的问题与上述所有问题都不同。

代码很好,但我正在使用 Sublime Text 并在 Windows 上的 sublime 文本内的终端模拟器中运行它。这阻止了扫描仪实际上允许我尝试输入一些东西。

我使用 cmd 运行该应用程序,它运行良好。我希望这可以帮助别人。

于 2013-07-09T02:21:08.883 回答