Как запустить фоновое задание в веб-приложении на сервлет?

Я использую Java, и я хочу постоянно поддерживать сервлет в моем приложении, но я не понимаю, как это сделать. Мой сервлет имеет метод, который ежедневно подсчитывает количество пользователей из базы данных, а также общее количество пользователей из всей базы данных. Поэтому я хочу, чтобы сервлет постоянно работал для этого.

Ответ 1

Ваша проблема в том, что вы неправильно поняли цель сервлета. Он намеревался действовать по HTTP-запросам, не более того. Вам нужна только фоновая задача, которая запускается один раз в день.

EJB доступен? Используйте @Schedule

Если ваша среда поддерживает EJB (то есть настоящий сервер Java EE, такой как WildFly, JBoss, TomEE, Payara, GlassFish и т.д.), Используйте вместо этого @Schedule. Вот несколько примеров:

@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.
    }

    @Schedule(hour="*", minute="*", second="*/5", persistent=false)
    public void someFiveSecondelyJob() {
        // Do your job here which should run every 5 seconds.
    }

} 

Да, это действительно все. Контейнер автоматически заберет и будет управлять им.

EJB недоступен? Используйте ScheduledExecutorService

Если ваша среда не поддерживает EJB (т.е. вы используете не настоящий сервер Java EE, а контейнерный сервлет-контейнер, такой как Tomcat, Jetty и т.д.), Используйте ScheduledExecutorService. Это может быть инициировано ServletContextListener. Вот пример запуска:

@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);
        scheduler.scheduleAtFixedRate(new SomeHourlyJob(), 0, 1, TimeUnit.HOURS);
        scheduler.scheduleAtFixedRate(new SomeQuarterlyJob(), 0, 15, TimeUnit.MINUTES);
        scheduler.scheduleAtFixedRate(new SomeFiveSecondelyJob(), 0, 5, TimeUnit.SECONDS);
    }

    @Override
    public void contextDestroyed(ServletContextEvent event) {
        scheduler.shutdownNow();
    }

}

Где рабочие классы выглядят так:

public class SomeDailyJob implements Runnable {

    @Override
    public void run() {
        // Do your daily job here.
    }

}
public class SomeHourlyJob implements Runnable {

    @Override
    public void run() {
        // Do your hourly job here.
    }

}
public class SomeQuarterlyJob implements Runnable {

    @Override
    public void run() {
        // Do your quarterly job here.
    }

}
public class SomeFiveSecondelyJob implements Runnable {

    @Override
    public void run() {
        // Do your quarterly job here.
    }

}

Никогда не думайте об использовании java.util.Timer/java.lang.Thread в среде на основе Java EE/Servlet

И последнее, но не менее важное: никогда напрямую не использовать java.util.Timer и/или java.lang.Thread в Java EE. Это рецепт неприятностей. Подробное объяснение можно найти в этом ответе, связанном с JSF, на тот же вопрос: Создание потоков в управляемом компоненте JSF для запланированных задач с использованием таймера.

Ответ 2

Я бы предложил использовать библиотеку типа кварца для выполнения задачи через регулярные промежутки времени. Что делает сервлет действительно? Он отправляет вам отчет?

Ответ 3

Реализуйте два класса и вызовите startTask() в main.

public void startTask()
{
    // Create a Runnable
    Runnable task = new Runnable() {
        public void run() {
            while (true) {
                runTask();
            }
        }
    };

    // Run the task in a background thread
    Thread backgroundThread = new Thread(task);
    // Terminate the running thread if the application exits
    backgroundThread.setDaemon(true);
    // Start the thread
    backgroundThread.start();
}

public void runTask()
{
    try {
        // do something...         
        Thread.sleep(1000);

    } catch (InterruptedException e) {
        e.printStackTrace();
    }
}

Ответ 5

В производственной системе, которая может иметь несколько не-jee контейнеров. Используйте anot-планировщик предприятия, например планировщик Quartz, который можно настроить для использования базы данных для задачи maamgememt.