자바의 가상 머신은 두 가지 종류의 이벤트를 받아 프로그램을 종료한다.
- 프로그램이 정상적으로 종료되거나 System.exit()
- 사용자 인터럽트(Ctrl+C)나 사용자 로그오프 또는 시스템 종료와 같은 시스템 전체 이벤트에 대한 응답 이벤트
프로그램이 정상 또는 비정상 종료하기 전에 특정 작업을 수행할 수 있도록 자바는 java.lang.Runtime.addShutdownHook(Thread t)을 제공한다. 종료될 때 수행할 내용을 정의한 Thread를 인자로 전달한다.
기본 예제
아래 결과는 “End”가 출력된 후, “Hook Run” 이 출력된다.
public class ShutdownHookTest {
static class HookThread extends Thread {
@Override
public void run() {
System.out.println("Hook Run");
}
}
public static void main(String[] args) {
Runtime.getRuntime().addShutdownHook(new HookThread());
System.out.println("End");
}
}
아래처럼 비정상 동작으로 프로그램이 종료되어 “End”는 출력되지 않지만 “Hook Run”이 출력된다.
public class ShutdownHookTest {
static class HookThread extends Thread {
@Override
public void run() {
System.out.println("Hook Run");
}
}
public static void main(String[] args) {
Runtime.getRuntime().addShutdownHook(new HookThread());
int errorNum = 1 / 0;
System.out.println("End");
}
}
콘솔에서 sleep 동안 인터럽트(Ctrl+C)를 줄 경우에도 “End”는 출력되지 않지만 “Hook Run”이 출력된다.
public class ShutdownHookTest {
static class HookThread extends Thread {
@Override
public void run() {
System.out.println("Hook Run");
}
}
public static void main(String[] args) {
Runtime.getRuntime().addShutdownHook(new HookThread());
try {
System.out.println("sleep 3s");
Thread.sleep(3000);
} catch (Exception e) {
e.printStackTrace();
}
System.out.println("End");
}
}
순서 지정 불가능
Shutdown Hook은 실행되는 순서를 결정할 수 없다. 즉 addShutdownHook() 메소드끼리의 호출 순서에 상관없이 동시에 실행된다. 아래처럼 HookThread를 먼저 등록하고, HookThread2를 등록하였지만 실행할 때마다 출력되는 순서는 랜덤 하다.
public class ShutdownHookTest {
static class HookThread extends Thread {
@Override
public void run() {
System.out.println("Hook Run1");
}
}
static class HookThread2 extends Thread {
@Override
public void run() {
System.out.println("Hook Run2");
}
}
public static void main(String[] args) {
Runtime.getRuntime().addShutdownHook(new HookThread());
Runtime.getRuntime().addShutdownHook(new HookThread2());
System.out.println("End");
}
}
Hook의 실행 막기
Runtime.halt(int)를 사용하면 등록된 Shutdown Hook을 실행하지 않고 종료된다. 만약 종료 중에 Halt가 호출될 경우 (종료 중이라면 등록된 Shutdown Hook 쓰레드들이 동시에 실행되는 상태) 실행 중인 Hook이 마칠 때까지 대기하지 않고, 바로 종료한다. 아래 예제를 보면 종료되기 전에 halt() 메소드를 호출하였으므로 등록된 Hook이 실행되지 않는다.
// ...
public static void main(String[] args) {
Runtime.getRuntime().addShutdownHook(new HookThread());
Runtime.getRuntime().addShutdownHook(new HookThread2());
System.out.println("End");
Runtime.getRuntime().halt(0);
}
}
Hook 발생 중 Hook 제어 불가능
Shutdown Hook이 시작되면 Hook을 추가할 수도 제거할 수 없다. 시도할 경우 IllegalStateException 예외가 발생한다. 아래 예제는 예외가 발생하는 경우이다. Shutdown Hook에 추가한 Hook Thread에 새로운 Hook을 추가하였다. 따라서 종료되면서 IllegalStateException 예외가 발생한다.
public class ShutdownHookTest {
static class HookThread extends Thread {
@Override
public void run() {
Runtime.getRuntime().addShutdownHook(new HookThread());
System.out.println("Hook Run1");
}
}
// ...
}
이외 유의사항
쓰레드는 언제나 어렵다. 복잡하고 디버깅하기도 힘들다. Shutdown Hook도 마찬가지로 쓰레드를 사용하므로 제대로 구현하지 않는다면 원하는 대로 작동하지 않을 수 있다.
프로그램이 종료될 때 모든 쓰레드 Hook이 동시에 실행되며 이외에 쓰레드들도 실행 중일 것이다. 따라서 thread-safe 하게 작성하여 데드락을 피해야 한다.
뿐만 아니라, 무조건 실행된다는 보장이 없으므로 Shutdown Hook 메소드를 100% 신뢰하지는 않는 것이 좋다. 정말 중요한 작업이라면 명시적인 종료자 메소드를 구현하여 사용하는 게 안전할 것이다.
마지막으로 Hook 쓰레드에서 많은 시간을 소모하면 안 된다. 만약 시스템을 종료하는 상황이라면 시스템이 종료되면 JVM이 종료되면서 Hook이 실행되겠지만 시간이 오래 걸릴 경우, 시스템은 무시하고 강제로 종료돼버린다.
'Java > 기본' 카테고리의 다른 글
[JAVA] Marker Interface : 아무것도 선언되지 않은 인터페이스 (0) | 2021.12.28 |
---|---|
[JAVA] Constant Interface : 상수를 정의한 인터페이스 (1) | 2021.12.28 |
[JAVA][Weblogic] java.lang.NoClassDefFoundError: Could not initialize class org.apache.poi.POIXMLDocument (0) | 2021.11.15 |
[JAVA] HEX String to bytes (0) | 2021.09.28 |
[JAVA] CountDownLatch : 다른 쓰레드를 기다리는 방법 (0) | 2021.09.16 |