从spring mvc app提交后台任务

时间:2012-10-01 09:33:37

标签: java multithreading spring-mvc

我已经开始使用spring MVC应用程序,我接下来要做的就是从我的应用程序启动或提交后台任务。

基本上,即使用户决定在应用上做其他事情,我也会继续执行任务,直到完成为止。

但是如果需要,我还想停止/终止/暂停任务。因为在我寻找一个好的/更好的方法之前我没有这样做过。

我发现这些很有用:

http://blog.springsource.com/2010/01/05/task-scheduling-simplifications-in-spring-3-0/

How do you kill a thread in Java?

Java threads: Is it possible view/pause/kill a particular thread from a different java program running on the same JVM?

所以我想使用@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时,流程没有停止,我在这里缺少什么?

1 个答案:

答案 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()来电。