강의

멘토링

커뮤니티

인프런 커뮤니티 질문&답변

dltkdcksqkqh님의 프로필 이미지
dltkdcksqkqh

작성한 질문수

김영한의 실전 자바 - 고급 1편, 멀티스레드와 동시성

문제와 풀이

Future cancel기능을 사용했지만 interrupt가 발생하지 않을 때 어떻게 처리해야하나요?

작성

·

21

0

[질문 템플릿]
1. 강의 내용과 관련된 질문인가요? (예)
2. 인프런의 질문 게시판과 자주 하는 질문에 없는 내용인가요? (예)
3. 질문 잘하기 메뉴얼을 읽어보셨나요? (예)

[질문 내용]
안녕하세요.
Futurecancel기능에 대해 실험하면서 해결하기 어려운 부분을 만나 질문드립니다.

📋실험 내용

  1. Callablecall메서드 내부에서 I/O와 같은 블로킹 작업 진행(or Interrupt가 되지 않는 작업)

  2. Futurecancel 진행

  3. 스레드 호출 대상은 CancellationException 예외 발생 후 진행
    스레드 풀의 스레드는 계속 실행 중임을 확인

 

💥문제 사항

이런 방식으로 로직이 지속 실행되면 스레드 풀의 스레드는 종료되지 않으면서 고갈되어 가고 결국엔 작업이 쌓이다가 에러를 뱉어내고 오작동 하게 될 것이라 추측됩니다.

 

질문 내용

Q. cancel기능은 Interrupt 를 발생해주면서 작업을 중단합니다. 만약 I/O 작업 처럼 Interrupt가 먹히지 않는 로직에서는 어떻게 스레드의 실행을 종료하고 스레드 풀로 반납할 수 있을까요?

💻 코드

public class FutureCancelMain {
    private static boolean mayInterrupIfRunning = true;

    public static void main(String[] args) {
        ExecutorService es = Executors.newFixedThreadPool(1);
        Future<String> future = es.submit(new MyTask());
        log("Future.state: " + future.state());

        sleep(3000);

        log("future.cancel(" + mayInterrupIfRunning + ") 호출");
        boolean cancelResult = future.cancel(mayInterrupIfRunning);
        log("cancel(" + mayInterrupIfRunning + ") result: " + cancelResult);

        try {
            log("Future result: " + future.get());
        } catch (CancellationException e) {
            log("Future는 이미 취소되었습니다.");
        } catch (ExecutionException | InterruptedException e) {
            e.printStackTrace();
        }
        es.close();
    }

    static class MyTask implements Callable<String> {

        @Override
        public String call(){
            boolean flag = true;
//            Scanner scanner = new Scanner(System.in);
            while(flag) {
                log("무한 작업 중...");
            }
            return "Completed";
        }
    }

 

 

답변 1

0

안녕하세요, 인프런 AI 인턴입니다. dltkdcksqkqh님.

Futurecancel 기능을 어떤 식으로 사용해야 하는지에 대한 질문을 주셨네요. Future.cancel(true)는 해당 callable이 실행 중인 경우 인터럽트를 발생시켜 스레드를 중단시키려고 하지만, I/O 작업과 같은 블로킹 작업에서는 인터럽트가 즉시 작동하지 않을 수 있습니다. 이와 관련된 문제는 코드를 설계할 때 비동기 작업을 효율적으로 관리하는 것이 중요하다는 점을 시사합니다.

스레드 관리와 인터럽트 처리

  • 인터럽트 가능한 작업으로 제한하기: 수행하는 작업이 가능하면 Interrupt를 통해 중단될 수 있도록 설계하는 것이 좋습니다. I/O 작업 중에서도 일부는 NIO(New I/O)의 Channel을 사용하여 인터럽트가 가능한 상태로 변환할 수 있습니다.

  • 중단 플래그 사용하기: 스레드의 중단 여부를 인지하고 블로킹 상태를 피하기 위해 플래그를 사용할 수 있습니다. 이 방법은 긴 작업이 블록 중인지 확인할 수 있는 경우 적합합니다. 즉, 주기적으로 작업을 체크하고 인터럽트가 발생하지 않는 경우를 대비해 중단 플래그를 삽입하는 것입니다.

  • 자주 체크하는 루프 대기 추가하기: 길게 블로킹되는 작업 앞에 주기적으로 인터럽트를 판단할 수 있도록 Thread.sleep()이나 다른 대기 코드를 추가하는 방법이 있습니다.

@Override
public String call() {
    boolean flag = true;
    while (flag) {
        if (Thread.currentThread().isInterrupted()) {
            log("인터럽트 되었습니다. 종료...");
            break;
        }
        log("무한 작업 중...");
        try {
            Thread.sleep(100); // 주기적으로 인터럽트를 체크
        } catch (InterruptedException e) {
            log("슬립 중 인터럽트 되었습니다. 종료...");
            Thread.currentThread().interrupt(); // 현재 스레드의 인터럽트 상태를 복원
            break;
        }
    }
    return "Completed";
}

추가적으로 참고할 수 있는 링크

위 링크에서는 스레드가 어떻게 인터럽트에 반응하는지에 대한 논의가 이뤄졌습니다. 스레드의 인터럽트 처리와 작업의 중단 방식에 대한 자세한 내용을 파악하는 데 도움이 될 것입니다.

위에서 설명드린 방법들이 실질적인 문제 해결에 기여했으면 좋겠습니다. 질문자님의 상황에 맞는 효율적인 설계를 통해 문제를 해결하시기 바랍니다.

저는 질문자님의 이해를 돕고자 지식공유자님의 기존 답변을 참고하여 유사한 답변 링크를 추천해드리고 있습니다.
현재 베타 기능으로 답변이 만족스럽지 않을 수 있는 점 양해 부탁드립니다. 🙏
추가적으로 궁금한 점이 있으시면, 이어서 질문해 주세요. 곧 지식공유자께서 답변해 주실 것입니다.

dltkdcksqkqh님의 프로필 이미지
dltkdcksqkqh

작성한 질문수

질문하기