Как запустить фоновую задачу в веб-приложении на основе сервлетов?
Вопрос
Я использую Java, и я хочу, чтобы сервлет постоянно работал в моем приложении, но я не понимаю, как это сделать.В моем сервлете есть метод, который ежедневно выдает количество пользователей из базы данных, а также общее количество пользователей из всей базы данных.Поэтому я хочу, чтобы сервлет постоянно работал для этого.
Решение
Ваша проблема в том, что вы неправильно понимаете цель сервис. Анкет Он намерен действовать по 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, а сервис Barebones, такой как 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 для запланированных задач с помощью таймера.
Другие советы
Я бы предложил использовать библиотеку типа quartz для выполнения задачи через регулярные промежутки времени.Что на самом деле делает сервлет ?Он отправляет вам отчет ?
Внедрить два класса и позвонить 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();
}
}
Вы можете использовать Cron4j. http://www.sauronsoftware.it/projects/cron4j/manual.php
В производственной системе, в которой могут быть несколько контейнеров без JEE. Используйте Anot Enterprise Scheduler, такой как Quartz Scheduler, который можно настроить для использования базы данных для задачи Maamgememt.