我想知道是否可以在应用程序范围内的Bean中使用Timer
。
例如,假设我要创建一个计时器任务,该任务每天向每个注册成员发送一堆电子邮件。我正在尝试使用尽可能多的JSF,我想知道这是否可以接受(我知道听起来有些奇怪)。
到目前为止,我已经在ServletContextListener
中使用了上述所有内容。 (我不想使用任何应用程序服务器或cron作业,我想保留
我的网络应用程序中的上述内容。)
是否有一个聪明的JSF方法可以做到这一点,还是我应该坚持旧的模式?
最佳答案
介绍
至于从JSF托管Bean内生成线程,仅当您希望能够通过#{managedBeanName}
在 View 中或在其他托管Bean中通过 @ManagedProperty("#{managedBeanName}")
引用它时才有意义。您仅应确保实现 @PreDestroy
,以确保每当Webapp即将关闭时,所有这些线程都被关闭,就像在 contextDestroyed()
的 ServletContextListener
方法中所做的那样(是的,您这样做了吗?)。另请参阅Is it safe to start a new thread in a JSF managed bean?
切勿在Java EE中使用java.util.Timer
至于在JSF托管bean中使用java.util.Timer
,则绝对不应该使用老式Timer
来使用,而是绝对使用现代 ScheduledExecutorService
来使用。 Timer
具有以下主要问题,使其不适合在长时间运行的Java EE Web应用程序中使用(引自Java Concurrency in Practice):
Timer
对系统时钟的变化很敏感,而ScheduledExecutorService
不敏感。 Timer
只有一个执行线程,因此长时间运行的任务可能会延迟其他任务。 ScheduledExecutorService
可以配置任何数量的线程。 TimerTask
中引发的任何运行时异常都会杀死一个线程,从而使Timer
失效,即计划的任务将不再运行。 ScheduledThreadExecutor
不仅可以捕获运行时异常,还可以根据需要处理它们。抛出异常的任务将被取消,但其他任务将继续运行。 除了书名外,我还可以想到其他缺点:
cancel()
进行Timer
,那么在取消部署后它将继续运行。因此,在重新部署新线程之后,将再次执行相同的工作。 Etcetera。到目前为止,它已经变成了“一劳永逸”,您无法通过编程方式将其取消。您基本上需要关闭并重新启动整个服务器以清除以前的线程。 Timer
线程未标记为守护程序线程,则它将阻止Webapp的取消部署和服务器的关闭。基本上,您需要强行终止服务器。主要缺点是该Web应用程序将无法通过以下方式执行正常的清除操作: contextDestroyed()
和@PreDestroy
方法。 EJB可用吗?使用
@Schedule
如果您以Java EE 6或更高版本为目标(例如JBoss AS,GlassFish,TomEE等,因此不是像Tomcat这样的准系统JSP / Servlet容器),请改为使用
@Singleton
EJB和 @Schedule
方法。这样,容器将担心通过ScheduledExecutorService
池化和销毁线程。然后,您只需要以下EJB:@Singleton
public class BackgroundJobManager {
@Schedule(hour="0", minute="0", second="0", persistent=false)
public void someDailyJob() {
// Do your job here which should run every start of day.
}
@Schedule(hour="*/1", minute="0", second="0", persistent=false)
public void someHourlyJob() {
// Do your job here which should run every hour of day.
}
@Schedule(hour="*", minute="*/15", second="0", persistent=false)
public void someQuarterlyJob() {
// Do your job here which should run every 15 minute of hour.
}
}
如果有必要,这可以通过
@EJB
在托管bean中使用:@EJB
private BackgroundJobManager backgroundJobManager;
EJB不可用?使用
ScheduledExecutorService
如果没有EJB,则需要手动使用
ScheduledExecutorService
。应用程序范围的托管bean实现看起来像这样:@ManagedBean(eager=true)
@ApplicationScoped
public class BackgroundJobManager {
private ScheduledExecutorService scheduler;
@PostConstruct
public void init() {
scheduler = Executors.newSingleThreadScheduledExecutor();
scheduler.scheduleAtFixedRate(new SomeDailyJob(), 0, 1, TimeUnit.DAYS);
}
@PreDestroy
public void destroy() {
scheduler.shutdownNow();
}
}
SomeDailyJob
如下所示:public class SomeDailyJob implements Runnable {
@Override
public void run() {
// Do your job here.
}
}
如果根本不需要在 View 或其他托管Bean中引用它,那么最好使用
ServletContextListener
使其与JSF分离。@WebListener
public class BackgroundJobManager implements ServletContextListener {
private ScheduledExecutorService scheduler;
@Override
public void contextInitialized(ServletContextEvent event) {
scheduler = Executors.newSingleThreadScheduledExecutor();
scheduler.scheduleAtFixedRate(new SomeDailyJob(), 0, 1, TimeUnit.DAYS);
}
@Override
public void contextDestroyed(ServletContextEvent event) {
scheduler.shutdownNow();
}
}
关于jsf - 使用计时器在JSF托管Bean中生成线程以执行计划的任务,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/7499534/