Java 예약 작업 프레임워크

함께 쓰는 습관을 들이세요! "너겟 데일리 뉴 플랜 · 4월 업데이트 챌린지" 참여 5일차입니다. 클릭하시면 이벤트 내용을 보실 수 있습니다 .

머리말

시스템을 개발하는 과정에서 다음 요구 사항이 있습니다.

  • 아침에 주문 통계 보고서 생성
  • 정기적으로 기사 푸시 및 이메일 보내기
  • 5분마다 데이터 업데이트를 동적으로 가져옵니다.
  • 매일 밤 사용자의 일일 수입을 계산하고 사용자에게 푸시
  • .....

일반적으로 이러한 요구 사항은 시간 지정 작업을 통해 달성되며 Java의 몇 가지 일반적인 시간 지정 작업 프레임워크가 나열됩니다.

예약된 작업 프레임워크

사진.png

타임태스크

Java를 배우기 시작한 이후로 TimeTask는 Timed Task를 처음 구현할 때 사용되며 Timer는 Timer 작업을 저장하기 위해 TaskQueue 클래스를 사용하며 최소 힙 구현을 기반으로 하는 우선순위 큐입니다. TaskQueue는 다음 실행 시간부터 작업 거리의 크기에 따라 작업을 정렬하여 힙 상단에 있는 작업이 먼저 실행되도록 합니다.

예제 코드:

 public static void main(String[] args)
    {
          TimerTask task = new TimerTask() {
              public void run() {
                  System.out.println("当前时间: " + new Date() + "n" +
                          "线程名称: " + Thread.currentThread().getName());
              }
          };
          Timer timer = new Timer("Timer");
          long delay = 5000L;
          timer.schedule(task, delay);
          
          System.out.println("当前时间: " + new Date() + "n" +
                  "线程名称: " + Thread.currentThread().getName());
    }
复制代码

작업 결과:

当前时间: Wed Apr 06 22:05:04 CST 2022n线程名称: main
当前时间: Wed Apr 06 22:05:09 CST 2022n线程名称: Timer

复制代码

예약된 작업이 5초 후에 실행되는 것을 결과를 통해 알 수 있습니다.

결점:

  • TimeTask 실행 태스크는 순차적으로만 수행할 수 있으며, 한 태스크가 장기간 실행되면 다른 태스크의 실행에 영향을 미칩니다.
  • 작업 실행 중에 예외가 발생하면 작업이 직접 중지됩니다.

시간이 지남에 따라 자바 기술은 지속적으로 업데이트되는데 TimeTask의 부족으로 인해 ScheduledExecutorService가 TimeTask를 대체할 것으로 보입니다.

ScheduledExecutorService

ScheduledExecutorService는 여러 구현 클래스가 있는 인터페이스이며 가장 일반적으로 사용되는 것은 ScheduledThreadPoolExecutor입니다. ScheduledThreadPoolExecutor 자체는 내부적으로 DelayQueue를 작업 대기열로 사용하고 작업의 동시 실행을 지원하는 스레드 풀입니다.

예제 코드:

 public static void main(String[] args) throws InterruptedException
   {
      ScheduledExecutorService executorService =
              Executors.newScheduledThreadPool(3);
      // 执行任务: 每 10秒执行一次
      executorService.scheduleAtFixedRate(() -> {
          System.out.println("执行任务:" + new Date()+",线程名称: " + Thread.currentThread().getName());
      }, 1, 10, TimeUnit.SECONDS);
    }
复制代码

결점:

  • jdk 자체 스레드 풀에서 사용하는 큐가 상대적으로 크고 OOM이 발생하기 쉽기 때문에 Executors 메소드를 사용하여 스레드 풀을 생성하지 마십시오.

  • Timed 작업은 JVM 단일 머신의 메모리를 기반으로 하며 Timed 작업이 다시 시작되면 사라집니다.

  • 풍부한 타이밍 작업을 달성하기 위해 cron 표현식을 지원할 수 없습니다.

스프링 태스크

Spring을 배운 후 Spring과 함께 제공되는 Task를 사용하기 시작했습니다. Spring Framework는 자체 예약된 작업과 함께 제공되며 다양한 예약된 작업 구성을 구현하기 위한 cron 표현식을 제공합니다.

예제 코드:

@EnableScheduling
@Component
public class SpringTask
{
    private Logger logger = LoggerFactory.getLogger(SpringTask.class);
    private static final SimpleDateFormat dateFormat = new SimpleDateFormat(
            "HH:mm:ss");

    /**
     * fixedRate:固定速率执行。每5秒执行一次。
     */
    @Scheduled(fixedRate = 5000)
    public void invokeTaskWithFixedRate()
    {
        logger.info("Fixed Rate Task :  Current Time  is  {}",
                dateFormat.format(new Date()));
    }

    /**
     * fixedDelay:固定延迟执行。距离上一次调用成功后2秒才执。
     */
    @Scheduled(fixedDelay = 2000)
    public void invokeTaskWithFixedDelay()
    {
        try
        {
            TimeUnit.SECONDS.sleep(3);
            logger.info("Fixed Delay Task : Current Time  is  {}",
                    dateFormat.format(new Date()));
        }
        catch (InterruptedException e)
        {
            logger.error("invoke task error",e);
        }
    }

    /**
     * initialDelay:初始延迟。任务的第一次执行将延迟5秒,然后将以5秒的固定间隔执行。
     */
    @Scheduled(initialDelay = 5000, fixedRate = 5000)
    public void invokeTaskWithInitialDelay()
    {
        logger.info("Task with Initial Delay : Current Time is  {}",
                dateFormat.format(new Date()));
    }

    /**
     * cron:使用Cron表达式,每隔5秒执行一次
     */
    @Scheduled(cron = "0/5 * * * * ? ")
    public void invokeTaskWithCronExpression()
    {
        logger.info("Task Cron Expression:  Current Time  is  {}",
                dateFormat.format(new Date()));
    }
}

复制代码

결과:

2022-04-06 23:06:20.945  INFO 14604 --- [   scheduling-1] com.fw.task.SpringTask                   : Task Cron Expression:  Current Time  is  23:06:20
2022-04-06 23:06:22.557  INFO 14604 --- [   scheduling-1] com.fw.task.SpringTask                   : Task with Initial Delay : Current Time is  23:06:22
2022-04-06 23:06:22.557  INFO 14604 --- [   scheduling-1] com.fw.task.SpringTask                   : Fixed Rate Task :  Current Time  is  23:06:22
2022-04-06 23:06:25.955  INFO 14604 --- [   scheduling-1] com.fw.task.SpringTask                   : Fixed Delay Task : Current Time  is  23:06:25
2022-04-06 23:06:25.955  INFO 14604 --- [   scheduling-1] com.fw.task.SpringTask                   : Task Cron Expression:  Current Time  is  23:06:25
2022-04-06 23:06:27.555  INFO 14604 --- [   scheduling-1] com.fw.task.SpringTask                   : Task with Initial Delay : Current Time is  23:06:27
2022-04-06 23:06:27.556  INFO 14604 --- [   scheduling-1] com.fw.task.SpringTask                   : Fixed Rate Task :  Current Time  is  23:06:27

复制代码

@EnableScheduling은 예약된 작업을 활성화해야 하며 @Scheduled(cron = "0/5 * * * * ?")는 예약된 작업에 대한 규칙을 구성합니다. cron 표현식은 풍부한 시간 지정 작업 구성을 지원합니다. 익숙하지 않은 경우 cron.qqe2.com/ 을 확인할 수 있습니다.

이점:

간단하고 사용하기 편리하며 다양한 복잡한 타이밍 작업 구성 지원

결점:

  • 독립형 형식을 기반으로 하는 타이밍 작업은 다시 시작되면 타이밍 작업이 사라집니다.

  • 기본적으로 예약된 작업은 단일 스레드 실행 작업이며 병렬 실행이 필요한 경우 @EnableAsync를 활성화해야 합니다.

  • 통합된 그래픽 작업 스케줄링 관리가 없으며 시간이 지정된 작업을 제어할 수 없습니다.

발문

모든 기술의 출현에는 그 가치가 있으며 최고의 기술은 없으며 가장 적합한 기술만 있습니다. 우리는 다양한 요구 사항에 가장 적합한 기술을 선택해야 하며 과도한 아키텍처를 사용하지 않습니다. 이 기사에서는 독립 실행형 환경에서 시간 지정 작업을 구현하는 기술에 대해 설명하고 다음 기사에서는 분산 작업 프레임워크에 대해 설명하기 시작합니다.

추천

출처juejin.im/post/7083510602441687054