我已经开始使用spring MVC应用程序,我接下来要做的就是从我的应用程序启动或提交后台任务。
基本上,即使用户决定在应用上做其他事情,我也会继续执行任务,直到完成为止。
但是如果需要,我还想停止/终止/暂停任务。因为在我寻找一个好的/更好的方法之前我没有这样做过。
我发现这些很有用:
http://blog.springsource.com/2010/01/05/task-scheduling-simplifications-in-spring-3-0/
How do you kill a thread in Java?
所以我想使用@Async任务来提交我的后台任务,但是想要稍后使用线程的id获取它并在需要时停止它?
这是正确的做法吗?我对多线程没有任何经验,所以我在这里听。
代码更新:
public interface Worker {
public void work();
public void cancel();
}
实施:
@Component("asyncWorker")
public class AsyncWorker implements Worker {
@Async
public void work() {
String threadName = Thread.currentThread().getName();
System.out.println(" " + threadName + " beginning work");
try {
Thread.sleep(10000); // simulates work
} catch (InterruptedException e) {
System.out.println("I stopped");
}
System.out.println(" " + threadName + " completed work");
}
public void cancel() { Thread.currentThread().interrupt(); }
}
用于测试目的的控制器:
@ResponseBody
@RequestMapping("/job/start")
public String start() {
asyncWorker.work();
return "start";
}
@ResponseBody
@RequestMapping("/job/stop")
public String stop() {
asyncWorker.cancel();
return "stop";
}
当我访问/job/start
时,我无法同时执行多个任务。另一个只在第一个完成后才开始执行
当我访问/job/stop
时,流程没有停止,我在这里缺少什么?
答案 0 :(得分:10)
使用线程ID太低而且脆弱。如果您决定使用@Async
annotation(不错的选择),则可以使用Future<T>
来控制任务执行。基本上,您的方法应该返回Future<T>
而不是void
:
@Async
public Future<Work> work() //...
现在您可以cancel()
Future
或等待它完成:
@ResponseBody
@RequestMapping("/job/start")
public String start() {
Future<Work> future = asyncWorker.work();
//store future somewhere
return "start";
}
@ResponseBody
@RequestMapping("/job/stop")
public String stop() {
future.cancel();
return "stop";
}
棘手的部分是以某种方式存储返回的future
对象,以便后续请求可用。当然,您不能使用字段或ThreadLocal
。您可以进行会话,但请注意Future
不可序列化,并且无法跨群集工作。
由于@Async
通常由线程池支持,因此您的任务甚至可能无法启动。取消只会将其从池中删除。如果任务已在运行,您可以isInterrupted()
主题标记或处理InterruptedException
来发现cancel()
来电。