基于控制台的应用程序不显示建议

时间:2017-07-11 13:55:48

标签: java console-application java.util.scanner

我正在开发基于控制台的Java应用程序。我要向用户显示选择数据库的建议。我使用Scanner来读取输入,并使用单独的线程来检查输入是否包含TAB以便打印建议。

更新

根据以下答案,我将synchronized块添加到代码中,并且Exception消失了。但是,我没有在控制台上看到任何建议。以下是我目前的代码:

import java.util.ArrayList;
import java.util.List;
import java.util.Scanner;

class Test {

    private static List<String> suggestions = new ArrayList<>();

    private static final Object lock = new Object();

    public static void main(String[] arguments) {

        suggestions.add("H2");
        suggestions.add("Mongo");
        suggestions.add("MySQL");
        suggestions.add("Oracle");
        suggestions.add("PostgreSQL");
        suggestions.add("SQLite");

        Scanner scanner = new Scanner(System.in);
        System.out.println("Enter a database name, press TAB for suggestions");

        new Thread(new Runnable() {
            public void run() {
                while (true) {
                    synchronized (lock) {
                        if (scanner.hasNext()) {
                            String input = scanner.next();
                            if (input.contains("\\t")) {
                                System.out.println(getSuggestions(input));
                            }
                        }
                    }
                }
            }
        }).start();

        synchronized (lock) {
            String selectedDatabase = scanner.nextLine();
            System.out.println(selectedDatabase);
        }

    }

    private static List<String> getSuggestions(String input) {
        List<String> possibleSuggestions = new ArrayList<>();
        for (String suggestion : suggestions) {
            if (suggestion.startsWith(input)) {
                possibleSuggestions.add(suggestion);
            }
        }
        return possibleSuggestions;
    }

}

有人可以帮忙吗?

1 个答案:

答案 0 :(得分:3)

简单地说,Scanner不是一个线程安全的类,你在两个不同的线程中使用它。

您将扫描程序在主线程中实例化,然后在另一个中使用它。在后台,Scanner的构造函数可能具有初始化的字段,这些字段不一定与另一个线程同步。

当另一个线程运行时,你在主线程中执行scanner.nextLine(),它可能与执行scanner.hasNext()的另一个线程完全同时执行,从而导致并发访问。

您需要一种方法来同步对扫描仪的访问(无处不在),例如:通过

synchronized (lock) {
    if (scanner.hasNext()) {
        String input = scanner.next();
        if (input.contains("\\t")) {
            System.out.println("ok");
        }
    }
}

其中lock是您同步的静态字段:

private static final Object lock = new Object();