首页 > 解决方案 > 如何连续扫描输入到控制台?

问题描述

我自己承担了学习Java的任务。我的想法是创建一个只有文本控制台的简单游戏。“AI”(计时器)会定期发送一个字符串,玩家必须写一个正确的字符串作为响应,否则他/她会失去生命。

因此,我的第一个问题是:有没有一种简单的方法可以将计时器和扫描仪结合起来?我需要它不断地“监视”控制台行中的字符串。

经过一段时间的搜索和尝试,我主要在扫描文本时生成或生成字符串时遇到了困难,我发现了以下代码,但它有一个问题:

if ((name =in.nextLine(2000)) ==null)

例如,如果我将条件重写为与 !="a" 而不是 null 进行比较,则代码将忽略该条件并始终写入“太慢了!” 无论。如果它是 =="a" 它总是说 Hello, a。我完全不明白为什么,它似乎忽略了逻辑。所以第二个问题是,为什么当它不同时它会忽略逻辑?我该如何解决?

public class TimedScanner
{
    public TimedScanner(InputStream input)
    {
        in = new Scanner(input);
    }

    private Scanner in;
    private ExecutorService ex = Executors.newSingleThreadExecutor(new ThreadFactory()
    {
        @Override
        public Thread newThread(Runnable r)
        {
            Thread t = new Thread(r);
            t.setDaemon(true);
            return t;
        }
    });

    public static void main(String[] args) {
        TimedScanner in = new TimedScanner(System.in);
        int playerHealth = 5;
        System.out.print("Enter your name: ");
        try {
            while (playerHealth > 0) {
                String name = null;
                if ((name = in.nextLine(3000)) ==null) {
                    System.out.println(name);
                    System.out.println("Too slow!");
                    playerHealth--;
                } else {
                    System.out.println(name);
                    System.out.println("Hello, " + name);
                }
            }
        } catch (InterruptedException | ExecutionException e) {
            //TODO Auto-generated catch block
            e.printStackTrace();
        }
    }

    public String nextLine(int timeout) throws InterruptedException, ExecutionException
    {
        Future<String> result = ex.submit(new Worker());
        try
        {
            return result.get(timeout, TimeUnit.MILLISECONDS);
        }
        catch (TimeoutException e)
        {
            return null;
        }
    }

    private class Worker implements Callable<String>
    {
        @Override
        public String call() throws Exception
        {
            return in.nextLine();
        }
    }
}

这是关于它应该做什么的非常简单的想法。在此期间,我计划放入一个随机选择的字符串,它将与控制台输入和错误输入进行比较 = playerHealth--; 正确输入其他内容。

标签: javatimerscanning

解决方案


2)为什么当它不同时它会忽略逻辑?我该如何解决?

你说过:

例如,如果我将条件重写为与!="a"而不是 null 进行比较,则代码将忽略该条件并始终写入“太慢了!” 无论。

在 Java 中,从不(或几乎从不)使用 == 或 != 比较两个字符串。字符串是一个对象,因此使用 == 比较它们意味着按地址而不是按值比较它们。所以

if ((name = in.nextLine(3000)) != "a")

将总是(或几乎总是)返回 true,因为从 in#nextLine 返回的任何字符串,无论是“a”还是不同的东西,都将在堆上分配到与硬编码的“a”字符串不同的地址。我说“几乎”的原因是因为 Java 使用字符串池的概念:当创建对文字的新引用时,它会检查字符串是否已经存在于池中以便重用它。但是你永远不应该依赖 ==。相反,使用Object.Equals()

更多关于 Java 字符串池的讨论在这里

1)有没有一种简单的方法来结合定时器和扫描仪?

好吧,控制台 UI 在读取用户输入时对多线程并不友好,但可以做到......

您的代码有一个问题:每当玩家失去一条生命时,它必须按两次 Enter - 当它连续失去 2 条生命时,它必须按 3 次 Enter 才能收到“AI”的积极反馈。这是因为您没有杀死前面的线程/取消前面的任务。所以我建议以下代码:

private static Scanner in;

public String nextLine(int timeout) throws InterruptedException, ExecutionException
{
    //keep a reference to the current worker
    Worker worker = new Worker();
    Future<String> result = ex.submit(worker);
    try
    {
        return result.get(timeout, TimeUnit.MILLISECONDS);
    }
    catch (TimeoutException e)
    {
        //ask the worker thread to stop
        worker.interrupt();
        return null;
    }
}

private class Worker implements Callable<String>
{
    //you want the most up-to-date value of the flag, so 'volatile', though it's not really necessary
    private volatile boolean interrupt;

    @Override
    public String call() throws Exception
    {
        //check whether there's something in the buffer;
        while (System.in.available() == 0){
            Thread.sleep(20);
            //check for the interrupt flag
            if(interrupt){
                throw new InterruptedException();
            }
        }
        //once this method is called there's no friendly way back - that's why we checked for nr of available bytes previously
        return in.nextLine();
    }

    public void interrupt(){
        this.interrupt = true;
    }
}

推荐阅读