ServiceImpl中的GWT,Google App Engine,TimerTask或Thread抛出异常

时间:2009-08-14 09:48:59

标签: java google-app-engine gwt multithreading

我正在使用GWT和Google App Engine。我有一系列记录,我想每30分钟更新一次。 在 ServiceImpl 中,我有以下代码:

new Timer().schedule(new TimerTask(){
    @Override
    public void run() {
        try {
            Thread.sleep(30000);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        result = updateFeeds();
    }
}, 30000,Long.MAX_VALUE);

当我运行应用程序时,我得到:

com.google.gwt.user.server.rpc.UnexpectedException:
Service method 'public abstract java.util.List org.joke.planet.client.FeedService.loadFeeds()' threw an unexpected exception:
java.security.AccessControlException: access denied (java.lang.RuntimePermission modifyThreadGroup)

在第一行或粘贴代码。

我的问题是如何让一个在GWT + AppEngine服务中工作的后台工作者?

5 个答案:

答案 0 :(得分:6)

你不能 - 后台工作者暗示一个线程,并且gae中的线程创建不起作用。

您的任务的答案不是创建线程或后台工作者,而是使用此http://code.google.com/appengine/docs/java/config/cron.html

答案 1 :(得分:3)

你先生,可以从appengine新的ThreadManager课程中受益。

https://developers.google.com/appengine/docs/java/backends/overview#background_threads

使用ThreadManager.currentRequestThreadFactory()获取一个Thread工厂,它将在当前请求完成后立即中断生成的线程,并使用ThreadManager.backgroundThreadFactory()生成将持续任意长度的线程(但仅适用于后端)

对于前端请求,建议跟踪线程,如果要在appengine的请求过滤器中断它们之前完成它们,请调用thread.join()。

答案 2 :(得分:3)

您不能使用java.util.Timer,因为它会创建一个普通的线程,这在AppEngine上是不允许的。但是,您可以使用AppEngine Modules API的后台线程工具,如下所述: https://developers.google.com/appengine/docs/java/modules/#Java_Background_threads

请注意,您只能在手动缩放的实例上使用后台线程。在此示例中,后台线程每30秒打印一条消息,永远:

Thread thread = ThreadManager.createBackgroundThread(new Runnable() {
  public void run() {
    try {
      while (true) {
        System.err.println("Doing something!");
        Thread.sleep(30_000);
      }
    } catch (InterruptedException ex) {
    }
  }
});
thread.start();

答案 3 :(得分:0)

另请参阅task queues作为替代方案。

答案 4 :(得分:-1)

问题不仅存在于App引擎中,而且通常存在于任何Servlet容器中。如果一个人在服务方法中(你总是在Servlet容器中),你就无法创建线程并进行休眠。

在今天的可扩展服务领域,thread.sleep是一件坏事......