gpt4 book ai didi

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

转载 作者:行者123 更新时间:2023-12-03 09:40:17 24 4
gpt4 key购买 nike

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

    24 4 0
    Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
    广告合作:1813099741@qq.com 6ren.com