jsf - 使用计时器在JSF托管Bean中生成线程以执行计划的任务

标签 jsf jsf-2 timer scheduled-tasks

我想知道是否可以在应用程序范围内的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/

    相关文章:

    java - 带内部 ui 的 Primefaces 数据表 :repeat

    jsf - 如何逃避:selectItem itemLabel attribute

    java - JSF 不解析 #{} 标签

    java - JSF 运行时页面名称传递

    richfaces - "no tag was found"错误

    java - JSF2 和 Spring AOP

    c++ - 计时器功能扭曲可拖动的线段

    c# - 为计时器间隔分配一个 double 值

    jsf - java.lang.ClassCastException : org. apache.xerces.jaxp.SAXParserFactoryImpl 无法转换为 javax.xml.parsers.SAXParserFactory

    java - 60 秒不活动后退出 Java 应用程序?