- html - 出于某种原因,IE8 对我的 Sass 文件中继承的 html5 CSS 不友好?
- JMeter 在响应断言中使用 span 标签的问题
- html - 在 :hover and :active? 上具有不同效果的 CSS 动画
- html - 相对于居中的 html 内容固定的 CSS 重复背景?
我需要让线程监听system.in,并连接所有输入,并且在执行某个命令或10秒钟没有输入时,它需要运行另一个将使用收集的信息的线程。
为了清楚起见,每次输入数据时,我都需要重置10秒。
作为并发编程的初学者,我不太确定如何解决这个问题
最佳答案
欢迎使用StackOverflow!
实现要求的一种简单方法是schedule扫描命令,然后等待相应的Future
返回结果或引发异常,同时等待一段时间。扫描命令是指Callable
,它将从System.in
变为scan the next line。
在这种情况下,您将不需要使用手工Thread
处理复杂的多线程。只需创建一个合适的ExecutorService
(通过使用Executors
类中的相应静态方法调用)来调度命令。 ExecutorService
就像Thread
的调度程序,即Thread
的池,用于处理它们的生命周期,并负责例如创建它们并启动它们。Future
是一个接口,它的一个实例可让您监视执行任务(例如Thread
)的执行时间,即检查是否完成,取消它等。Callable
是一个接口,实现其中的一个只是在计算/方法调用之后生成/返回结果,或者在无法生成结果的情况下抛出Exception
。 Future
在我们的上下文中将由ExecutorService
的调度命令返回,以让我们监视提交的Callable
的生命周期...
我们将要提交的Callable
只会返回Scanner.nextLine
方法调用的结果。通过将Callable
提交给scehduler,我们得到了一个Future
,它使Callable
在给定的时间内等待完成。为了无限期地等待Callable
的完成,我们使用了get
方法。为了等待特定的超时时间(这是我们要寻找的时间),我们使用了另一个get
方法,并为其提供了我们希望等待的时间。
我们可以通过ExecutorService
帮助器类(可以通过实例化创建它们)在Java 8(我正在使用,您可以从链接中得知)中创建几种类型的调度程序(即Executors
)。相应的类,但为简单起见,我们将使用Executors
静态方法)。我不是这些方面的专家,但总的来说,有一个固定的线程池,它最多可以在任何给定的时间运行给定数量的Thread
,有计划的线程池可以在以下时间执行Thread
基于时间的速率和周期,它们有单线程版本(即,相同的概念,一次仅一个Thread
),还有一个缓存的线程池,该线程池根据需要创建Thread
并重用现有的已完成线程,最后是一个工作窃取池,它的所有线程并行地阻塞/等待工作(我不确定最后一个线程,但是根据文档,当您的任务生成其他任务等时,这很有用)。
由于我们一次提交一个Callable
(一次调用一个Scanner.nextLine
),因此我们可以利用单线程版本。并且由于我们不在乎定期执行提交的Callable
,而是希望在每次完成后提交它,因此我们将使用固定的单线程池版本。
当用户的输入准备就绪时,您也不需要启动其他线程,但是您可以利用提交Callable
的相同线程。这是以下概念代码中的主线程:
import java.util.LinkedList;
import java.util.Objects;
import java.util.Scanner;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
public class Concept {
public static void main(final String[] args) {
final LinkedList<String> q = new LinkedList<>(); //The collection to hold all user's input.
final Scanner scan = new Scanner(System.in); //The Scanner of the System.in input stream.
final TimeUnit waitUnit = TimeUnit.SECONDS; //What unit of time should we use when waiting for input.
final long waitAmount = 10; //How much time (in 'waitUnit' units) should we wait for.
//An executor with a single (daemon) thread:
final ExecutorService scheduler = Executors.newSingleThreadExecutor(r -> {
final Thread t = new Thread(r);
t.setDaemon(true);
return t;
});
try {
try {
//Main loop for reading and waiting:
for (String input = scheduler.submit(() -> scan.nextLine()).get(waitAmount, waitUnit);
!Objects.equals(input, "stop");
input = scheduler.submit(() -> scan.nextLine()).get(waitAmount, waitUnit))
q.add(input); //Add the user's last input to the collection.
//If this is reached, then the user entered "stop" as input.
System.out.println("Ended by user's input.");
}
catch (final TimeoutException tx) {
//If this is reached, then the time timed out when waiting for user's input.
System.out.println("Ended by timeout.");
}
finally {
//Here you can "consume" however you like all the user's input from the collection:
q.forEach(line -> System.out.println(line)); //I'm just printing all of it.
}
}
catch (final InterruptedException | ExecutionException x) {
x.printStackTrace(); //This is where you handle unexpected exceptions.
}
finally {
//Whatever happened, don't forget to shutdown the ExecutorService:
scheduler.shutdown();
}
}
}
TimeoutException
,再次进行连接的用户输入的处理。
Executors
方法调用提供了
ThreadFactory
。
ThreadFactory
只是一个接口,其实现为给定的
Thread
创建
Runnable
。
Runnable
还是一个接口,这次定义了一个执行计算的单一方法(
run
)。在我们的案例中,此计算是在
ExecutorService
内部创建的,用于存储我们提交的
Callable
结果的引用,以便可用于返回的
get
的
Future
方法,该方法将进而使它可用于客户的代码。我为
ThreadFactory
提供的
ExecutorService
正在将每个
Thread
创建为一个守护程序。守护程序
Thread
不会终止程序。当所有非守护程序线程都完成后,程序将终止,而与其他(守护程序)线程是否仍在运行无关。
Callable
尚未完成。我们提交的
Callable
正在等待
System.in
的输入。这样该线程将无限期运行,或者直到用户输入内容为止。如果创建的
Thread
不是守护程序,则不会终止程序。这就是为什么我要使其成为守护程序。
System.in
对象的
Scanner
中读取内容呢?然后,您必须首先维护对最后一个
Future
方法调用返回的最后一个
ExecutorService.submit
的引用。
TimedCallable
的包装对象,您应该在每次扫描时使用该对象。即使超时或以“停止”字结束后,您也应继续使用它来扫描
System.in
:
import java.util.LinkedList;
import java.util.Objects;
import java.util.Scanner;
import java.util.concurrent.Callable;
import java.util.concurrent.ExecutionException;
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;
import java.util.concurrent.Future;
import java.util.concurrent.TimeUnit;
import java.util.concurrent.TimeoutException;
public class Main {
public static class TimedCallable<V> implements Callable<V> {
private final Callable<V> callable;
private final ExecutorService scheduler;
private Future<V> lastFuture;
public TimedCallable(final Callable<V> callable) {
this.callable = Objects.requireNonNull(callable);
scheduler = Executors.newSingleThreadExecutor(r -> {
final Thread t = new Thread(r);
t.setDaemon(true); //Needs to be a daemon in order to let the program end.
return t;
});
lastFuture = null;
}
@Override
public synchronized V call() throws InterruptedException, ExecutionException {
if (lastFuture == null)
try {
return callable.call();
}
catch (final Exception x) {
throw new ExecutionException(x);
}
final V v = lastFuture.get();
lastFuture = null;
return v;
}
public synchronized V call(final TimeUnit timeoutUnit,
final long timeoutAmount) throws TimeoutException, InterruptedException, ExecutionException {
if (lastFuture == null)
lastFuture = scheduler.submit(callable);
final V v = lastFuture.get(timeoutAmount, timeoutUnit); /*If it throws TimeoutException,
then the 'lastFuture' property will not be nulled by the following statement:*/
lastFuture = null;
return v;
}
}
public static void main(final String[] args) {
final LinkedList<String> q = new LinkedList<>(); //The collection to hold all user's input.
final Scanner scan = new Scanner(System.in); //The Scanner of the System.in input stream.
final TimeUnit waitUnit = TimeUnit.SECONDS; //What unit of time should we use when waiting for input.
final long waitAmount = 10; //How much time (in 'waitUnit' units) should we wait for.
//Instantiate the scanner's timed-callable:
final TimedCallable<String> scanNextLine = new TimedCallable<>(() -> scan.nextLine());
try {
try {
//Main loop for reading and waiting:
for (String input = scanNextLine.call(waitUnit, waitAmount); !Objects.equals(input, "stop"); input = scanNextLine.call(waitUnit, waitAmount))
q.add(input); //Add the user's last input to the collection.
//If this is reached, then the user entered "stop" as input.
System.out.println("Ended by user's input.");
}
catch (final TimeoutException tx) {
//If this is reached, then the time timed out when waiting for user's input.
System.out.println("Ended by timeout.");
}
finally {
//Here you can "consume" however you like all the user's input from the collection:
q.forEach(line -> System.out.println(line)); //I'm just printing all of it.
//Keep on using the Scanner via the TimedCallable:
System.out.println("Enter next line:");
System.out.println(scanNextLine.call());
System.out.println("Enter last line:");
System.out.println(scanNextLine.call());
}
}
catch (final InterruptedException | ExecutionException x) {
x.printStackTrace(); //This is where you handle unexpected exceptions.
}
}
}
关于java - 如果在过去15秒钟内在System.in上没有输入,如何使一个线程超时,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/60718858/
我正在编写一个具有以下签名的 Java 方法。 void Logger(Method method, Object[] args); 如果一个方法(例如 ABC() )调用此方法 Logger,它应该
我是 Java 新手。 我的问题是我的 Java 程序找不到我试图用作的图像文件一个 JButton。 (目前这段代码什么也没做,因为我只是得到了想要的外观第一的)。这是我的主课 代码: packag
好的,今天我在接受采访,我已经编写 Java 代码多年了。采访中说“Java 垃圾收集是一个棘手的问题,我有几个 friend 一直在努力弄清楚。你在这方面做得怎么样?”。她是想骗我吗?还是我的一生都
我的 friend 给了我一个谜语让我解开。它是这样的: There are 100 people. Each one of them, in his turn, does the following
如果我将使用 Java 5 代码的应用程序编译成字节码,生成的 .class 文件是否能够在 Java 1.4 下运行? 如果后者可以工作并且我正在尝试在我的 Java 1.4 应用程序中使用 Jav
有关于why Java doesn't support unsigned types的问题以及一些关于处理无符号类型的问题。我做了一些搜索,似乎 Scala 也不支持无符号数据类型。限制是Java和S
我只是想知道在一个 java 版本中生成的字节码是否可以在其他 java 版本上运行 最佳答案 通常,字节码无需修改即可在 较新 版本的 Java 上运行。它不会在旧版本上运行,除非您使用特殊参数 (
我有一个关于在命令提示符下执行 java 程序的基本问题。 在某些机器上我们需要指定 -cp 。 (类路径)同时执行java程序 (test为java文件名与.class文件存在于同一目录下) jav
我已经阅读 StackOverflow 有一段时间了,现在我才鼓起勇气提出问题。我今年 20 岁,目前在我的家乡(罗马尼亚克卢日-纳波卡)就读 IT 大学。足以介绍:D。 基本上,我有一家提供簿记应用
我有 public JSONObject parseXML(String xml) { JSONObject jsonObject = XML.toJSONObject(xml); r
我已经在 Java 中实现了带有动态类型的简单解释语言。不幸的是我遇到了以下问题。测试时如下代码: def main() { def ks = Map[[1, 2]].keySet()
一直提示输入 1 到 10 的数字 - 结果应将 st、rd、th 和 nd 添加到数字中。编写一个程序,提示用户输入 1 到 10 之间的任意整数,然后以序数形式显示该整数并附加后缀。 public
我有这个 DownloadFile.java 并按预期下载该文件: import java.io.*; import java.net.URL; public class DownloadFile {
我想在 GUI 上添加延迟。我放置了 2 个 for 循环,然后重新绘制了一个标签,但这 2 个 for 循环一个接一个地执行,并且标签被重新绘制到最后一个。 我能做什么? for(int i=0;
我正在对对象 Student 的列表项进行一些测试,但是我更喜欢在 java 类对象中创建硬编码列表,然后从那里提取数据,而不是连接到数据库并在结果集中选择记录。然而,自从我这样做以来已经很长时间了,
我知道对象创建分为三个部分: 声明 实例化 初始化 classA{} classB extends classA{} classA obj = new classB(1,1); 实例化 它必须使用
我有兴趣使用 GPRS 构建车辆跟踪系统。但是,我有一些问题要问以前做过此操作的人: GPRS 是最好的技术吗?人们意识到任何问题吗? 我计划使用 Java/Java EE - 有更好的技术吗? 如果
我可以通过递归方法反转数组,例如:数组={1,2,3,4,5} 数组结果={5,4,3,2,1}但我的结果是相同的数组,我不知道为什么,请帮助我。 public class Recursion { p
有这样的标准方式吗? 包括 Java源代码-测试代码- Ant 或 Maven联合单元持续集成(可能是巡航控制)ClearCase 版本控制工具部署到应用服务器 最后我希望有一个自动构建和集成环境。
我什至不知道这是否可能,我非常怀疑它是否可能,但如果可以,您能告诉我怎么做吗?我只是想知道如何从打印机打印一些文本。 有什么想法吗? 最佳答案 这里有更简单的事情。 import javax.swin
我是一名优秀的程序员,十分优秀!