Java - Timer, TimerTask 소개 및 예제

자바의 Timer는 어떤 작업이(TimerTask)가 특정 시간에 실행되도록 할 수 있습니다. 또는 일정 주기로 실행되도록 할 수 있습니다.

이 글에서는 Timer를 사용하여 어떤 작업을 일정 시간 지난 뒤에 처리하는 방법을 소개합니다.

  • 일정 시간이 지난 뒤에 Task 실행
  • 정해진 시간에 Task 실행
  • 일정 시간 간격으로(주기적으로) Task 실행
  • Timer 취소

일정 시간이 지난 뒤에 Task 실행

가장 간단하게 Timer를 사용하는 방법입니다. 다음과 같이 Timer와 TimerTask 객체를 생성합니다. Timer.schedule()에 실행시킬 Task와 Dealy(ms)를 인자로 전달하면, 그 시간이 지난 뒤에 Task가 한번 실행됩니다.

TimerTask task = new TimerTask() {
    public void run() {
        System.out.println(new Date() + " : Executing the task from "
                + Thread.currentThread().getName());
    }
};

Timer timer = new Timer("Timer");
long delay = 3000L;
System.out.println(new Date() + " : Scheduling....");
timer.schedule(task, delay);

Output:

Mon Nov 30 22:53:51 KST 2020 : Scheduling....
Mon Nov 30 22:53:54 KST 2020 : Executing the task from Timer

정해진 시간에 Task 실행

위의 예제와 비슷합니다. schedule()에 Delay 대신에 Date를 인자로 전달하면, 설정된 시간에 Task가 한번 실행됩니다.

TimerTask task = new TimerTask() {
    public void run() {
        System.out.println(new Date() + " : Executing the task from "
                + Thread.currentThread().getName());
    }
};

LocalDateTime localDateTime = LocalDateTime.now().plusSeconds(3);
Date date = Date.from(localDateTime.atZone(
        ZoneId.systemDefault()).toInstant());

Timer timer = new Timer("Timer");
System.out.println(new Date() + " : Scheduling....");
timer.schedule(task, date);

Output:

Mon Nov 30 22:55:49 KST 2020 : Scheduling....
Mon Nov 30 22:55:52 KST 2020 : Executing the task from Timer

일정 시간 간격으로(주기적으로) Task 실행

schedule()에 Dealy와 함께 Period를 인자로 전달할 수 있습니다. 그럼 Dealy 뒤에 Task가 실행되고, 일정 주기로 Task가 반복적으로 실행됩니다.

TimerTask task = new TimerTask() {
    public void run() {
        System.out.println(LocalDateTime.now() + " : Executing the task from "
                + Thread.currentThread().getName());
    }
};
Timer timer = new Timer("Timer");
long delay = 3000L;
long period = 1000L;
System.out.println(LocalDateTime.now() + " : Scheduling....");
timer.schedule(task, delay, period);

실행 결과를 보면 1초 간격으로 Task가 실행됩니다. 0.001ms씩 증가하고 있는데 Thread가 정확한 주기로 실행시키지 못하기 때문입니다.

> Task :TimerExample.main()
2020-11-30T23:01:56.504 : Scheduling....
2020-11-30T23:01:59.505 : Executing the task from Timer
2020-11-30T23:02:00.504 : Executing the task from Timer
2020-11-30T23:02:01.504 : Executing the task from Timer
....
2020-11-30T23:02:14.506 : Executing the task from Timer
2020-11-30T23:02:15.506 : Executing the task from Timer
2020-11-30T23:02:16.507 : Executing the task from Timer

고정된 주기로 실행

schedule()은 Task가 실행될 떄마다 0.001ms씩 증가할 떄가 있었습니다. 이것들이 누적되어 오차가 발생할 수 있습니다.

scheduleAtFixedRate()를 사용하면 고정된 주기로 Task를 실행하며, 이런 오차가 누적되지 않습니다.

TimerTask task = new TimerTask() {
    public void run() {
        System.out.println(LocalDateTime.now() + " : Executing the task from "
                + Thread.currentThread().getName());
    }
};
Timer timer = new Timer("Timer");
long delay = 3000L;
long period = 1000L;
System.out.println(LocalDateTime.now() + " : Scheduling....");
timer.scheduleAtFixedRate(task, delay, period);

Output:

11:07:04 PM: Executing task 'TimerExample.main()'...

> Task :compileJava
> Task :processResources NO-SOURCE
> Task :classes

> Task :TimerExample.main()
2020-11-30T23:07:07.541 : Scheduling....
2020-11-30T23:07:10.543 : Executing the task from Timer
2020-11-30T23:07:11.542 : Executing the task from Timer
2020-11-30T23:07:12.542 : Executing the task from Timer
....
2020-11-30T23:07:22.542 : Executing the task from Timer
2020-11-30T23:07:23.542 : Executing the task from Timer

Timer 취소

다음과 같이 TimerTask 안에서 cancel()을 호출하면 Timer가 취소됩니다.

TimerTask task = new TimerTask() {
    public void run() {
        System.out.println(new Date() + " : Executing the task from "
                + Thread.currentThread().getName());
        cancel();
    }
};
Timer timer = new Timer("Timer");
System.out.println(new Date() + " : Scheduling....");
timer.schedule(task, 1000, 1000);

Output:

Mon Nov 30 23:09:08 KST 2020 : Scheduling....
Mon Nov 30 23:09:09 KST 2020 : Executing the task from Timer

Timer 취소 (2)

또는 다음과 같이 TimerTask 외부에서 Timer.cancel()로 Timer를 취소할 수 있습니다.

TimerTask task = new TimerTask() {
    public void run() {
        System.out.println(new Date() + " : Executing the task from "
                + Thread.currentThread().getName());
    }
};
Timer timer = new Timer("Timer");
System.out.println(new Date() + " : Scheduling....");
timer.schedule(task, 1000, 1000);

Thread.sleep(5000);
System.out.println(new Date() + " : Canceling Timer....");
timer.cancel();

Output:

Mon Nov 30 23:10:17 KST 2020 : Scheduling....
Mon Nov 30 23:10:18 KST 2020 : Executing the task from Timer
Mon Nov 30 23:10:19 KST 2020 : Executing the task from Timer
Mon Nov 30 23:10:20 KST 2020 : Executing the task from Timer
Mon Nov 30 23:10:21 KST 2020 : Executing the task from Timer
Mon Nov 30 23:10:22 KST 2020 : Executing the task from Timer
Mon Nov 30 23:10:22 KST 2020 : Canceling Timer....
Loading script...

Related Posts

codechachaCopyright ©2019 codechacha