我正在使用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服务中工作的后台工作者?
答案 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是一件坏事......