ExecutorService close() 메소드 동작 설명 오류
[질문 템플릿]
1. 강의 내용과 관련된 질문인가요? 예
2. 인프런의 질문 게시판과 자주 하는 질문에 없는 내용인가요? 예
3. 질문 잘하기 메뉴얼을 읽어보셨나요? 예
[질문 내용]
ExecutorService 우아한 종료 - 구현
01:07 close() 메소드 설명 부분에
"shutdown() 을 호출하고, 하루를 기다려도 작업이 완료되지 않으면 shutdownNow() 를 호출한다." 에 오류가 있어 보입니다.
[내용 정정]
"shutdown() 을 호출하고, 작업이 완료될 때까지 무한정 반복대기한다. 인터럽트가 발생하면 shutdownNow() 를 호출한다." 로 변경해야 하지 않나요?
[근거]
Eclipse Temurin-21.0.4+7 (JDK 21) 버전 코드 원문
public interface ExecutorService implements Executor, AutoClosable {
// ..
@Override
default void close() {
boolean terminated = isTerminated();
if (!terminated) {
shutdown();
boolean interrupted = false;
// 작업 미완료시 재시도
while (!terminated) {
try {
// 하루간 작업 종료 대기, 작업 미완료시 `false` 반환
terminated = awaitTermination(1L, TimeUnit.DAYS);
} catch (InterruptedException e) {
if (!interrupted) {
shutdownNow();
interrupted = true;
}
}
}
if (interrupted) {
Thread.currentThread().interrupt();
}
}
}
}1일이 지났다고 해서 interrupt() 를 호출하지 않고 재시도 하는 것으로 보입니다.
[증명 코드]
MyExecutor 로 close() 메소드 대기 시간을 1일 -> 1초로 조정하여 오버라이드합니다.
10초간 대기하는 Task 를 실행시킵니다.
class ShutdownTest {
record CallableSleepTask(
String name,
long sleepMs
) implements Callable<Long> {
@Override
public Long call() throws Exception {
sleep(this.sleepMs);
return this.sleepMs;
}
}
static class MyExecutor extends ThreadPoolExecutor {
public MyExecutor() {
super(1, 1, 0, TimeUnit.SECONDS, new LinkedBlockingQueue<>());
}
@Override
public void close() {
boolean terminated = isTerminated();
if (!terminated) {
shutdown();
boolean interrupted = false;
while (!terminated) {
try {
// 종료 대기 시간 1초로 조정
terminated = awaitTermination(1L, TimeUnit.SECONDS);
} catch (InterruptedException e) {
if (!interrupted) {
shutdownNow();
interrupted = true;
}
}
}
if (interrupted) {
Thread.currentThread().interrupt();
}
}
}
}
@DisplayName("Should wait infinitely when .close()")
@Test
void executorServiceCloseTest() {
// given
try (ExecutorService myExecutor = new MyExecutor()) {
log("작업 시작");
CallableSleepTask sleepTask = new CallableSleepTask("test", 10_000L);
myExecutor.submit(sleepTask); // 10 초간 테스크 실행
log("작업 진행중.. close 대기");
// 기다리지 않고 .close() 자동 호출
}
}
}실행 결과
18:38:39.524 [ main] 작업 시작
18:38:39.527 [ main] 작업 진행중.. close 대기
(약 10초 대기 후 종료)
Process finished with exit code 0테스크 종료를 위해 10초 대기가 이뤄졌습니다.
close 가 조정된 대기시간인 1초 뒤에 인터럽트를 발생시켰다면 인터럽트 로그가 남았을 것이고, shutdownNow()가 호출됐다면 10초 대기없이 즉시 종료되었을 것입니다.
따라서 정정이 필요해보여요.
close 메소드는 "shutdown() 을 호출하고, 작업 완료시까지 무한정 반복대기한다. 인터럽트가 발생하면 shutdownNow() 를 호출한다."
답변 3
1
안녕하세요. milkcoke님
말씀하신 내용이 맞네요 🙂
코드로 검증까지 해주셔서 감사합니다.
1일마다 반복해서 스레드가 깨어나는 이유는 1일이라는 충분한 대기 시간을 통해 프로세스 성능에 영향을 최소화 하면서, 또 해당 스레드가 완전히 멈춘 것 처럼 보이지는 않도록 하기 위한 최소한의 장치로 보이네요.
다음 버전에 패치할게요!
0
안녕하세요, 인프런 AI 인턴입니다. milkcoke님.
ExecutorService의 close() 메소드에 대해 지적하신 부분과 함께 제시하신 증명 코드, 실행 결과 등을 보았습니다. 설명해주신 것처럼, close() 메소드는 shutdown()을 호출한 후, 작업이 모두 완료될 때까지 awaitTermination을 사용하여 무한정 대기하는 방식으로 설계되어 있습니다. 인터럽트가 발생하면 대기가 중단되고 shutdownNow()가 호출되게 되어 있죠.
이와 관련하여 유사한 질문과 답변을 아래 링크에서 참고할 수 있습니다. 이 링크에서는 ExecutorService의 명확한 종료 방법과 관련된 코드 동작에 대해 다루고 있습니다.
이 링크에서는 ExecutorService의 종료 방법에 대한 잘못된 이해로 인한 문제를 해결하면서, 정확한 종료 절차에 대한 설명을 제공하고 있습니다. 본 강의의 해당 부분이 잘못 설명되었다면, 강의를 제공하신 지식공유자께 제보하여 수정될 수 있도록 유도하는 것이 좋겠습니다.
저는 질문자님의 이해를 돕고자 지식공유자님의 기존 답변을 참고하여 유사한 답변 링크를 추천해드리고 있습니다. 현재 베타 기능으로 답변이 만족스럽지 않을 수 있는 점 양해 부탁드립니다. 🙏 추가적으로 궁금한 점이 있으시면, 이어서 질문해 주세요. 곧 지식공유자께서 답변해 주실 것입니다.
모니터와 synchronized, ReentrantLock, 원자적연산 CAS관련해서 추가적으로 더 깊게 공부했는데 제가 이해한 것이 맞나요??
0
84
1
Future cancel기능을 사용했지만 interrupt가 발생하지 않을 때 어떻게 처리해야하나요?
0
72
2
2.스레드 생성과 실행 - 17P 그림 문의
0
73
2
자바 동시성 현업에서 사용예시가 궁금합니다
0
133
1
ReentrantLock 동작 원리에 대한 질문입니다.
0
81
1
synchronized 대신 join 을 사용하여 순서를 보장할 경우 해법이 아닌가요?
0
93
2
메모리 가시성 문제 질문
0
69
1
Future4-이유 질문
0
64
1
오타 제보
0
74
2
runnable 질문
0
95
2
자바가 아닌 다른 언어에서도
0
111
1
메모리 가시성 문제가 발생하는 이유
0
125
1
데몬스레드가 어떨 때 사용이 되는지 궁금합니다.
0
78
1
Future.cancel() 취소를 사용하는 예가 궁금합니다.
0
84
1
synchronized 와 인스턴스의 lock
0
90
2
synchronized 키워드만으로 메모리 가시성이 해결되는 이유가 궁급합니다.
0
62
1
[스레드 풀과 Executor 프레임워크1][문제와 풀이] `log("일부 작업이 실패했습니다.");` 실행되는 경우가 있을까요?
0
59
1
원자적 연산이 멀티스레드 상황에서 안전한 이유를 제대로 이해했는지 확인 부탁드립니다
0
67
2
자바 고급 1,2,3편 같이 진행해도 문제없나요
0
77
2
CPU 캐시 때문에도 잔액이 200원이 될 수 있나요?
0
105
2
synchronized 메서드와 join 관련 질문이 있습니다.
0
69
2
섹션 14. 115번 강의 내용 질문
0
82
2
교안에 있는 7페이지 데이터 영역
0
51
1
Executor 스레드 풀 관리 질문
0
108
2





