我正在开发基于控制台的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;
}
}
有人可以帮忙吗?
答案 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();