자바의 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....
Related Posts
- Java - Unsupported class file major version 61 에러
- Java - String.matches()로 문자열 패턴 확인 및 다양한 예제 소개
- Java - 문자열 공백제거 (trim, replace)
- Java - replace()와 replaceAll()의 차이점
- Java - ArrayList 초기화, 4가지 방법
- Java - 배열 정렬(Sorting) (오름차순, 내림차순)
- Java - 문자열(String)을 비교하는 방법 (==, equals, compare)
- Java - StringBuilder 사용 방법, 예제
- Java - 로그 출력, 파일 저장 방법 (Logger 라이브러리)
- Java IllegalArgumentException 의미, 발생 이유
- Java - NullPointerException 원인, 해결 방법
- Seleninum의 ConnectionFailedException: Unable to establish websocket connection 해결
- Java - compareTo(), 객체 크기 비교
- Java - BufferedWriter로 파일 쓰기
- Java - BufferedReader로 파일 읽기
- Java charAt() 함수 알아보기
- Java - BigInteger 범위, 비교, 연산, 형변환
- Java contains()로 문자(대소문자 X) 포함 확인
- Java - Set(HashSet)를 배열로 변환
- Java - 문자열 첫번째 문자, 마지막 문자 확인
- Java - 문자열 한글자씩 자르기
- Java - 문자열 단어 개수 가져오기
- Java - 1초마다 반복 실행
- Java - 배열을 Set(HashSet)로 변환
- Java - 여러 Set(HashSet) 합치기
- Java - 명령행 인자 입력 받기
- Java - 리스트 역순으로 순회, 3가지 방법
- Java - 특정 조건으로 리스트 필터링, 3가지 방법
- Java - HashMap 모든 요소들의 합계, 평균 계산
- Java - 특정 조건으로 HashMap 필터링
- Java - 싱글톤(Singleton) 패턴 구현
- Java - 숫자 왼쪽에 0으로 채우기
- Java - String 배열 초기화 방법
- Java - 정렬된 순서로 Map(HashMap) 순회
- Java - HashMap에서 key, value 가져오기