BuNa_
IT Story
BuNa_
전체 방문자
오늘
어제
  • 분류 전체보기 (117)
    • CS (14)
      • 운영체제 (8)
      • 네트워크 (0)
      • Design Pattern (1)
      • OOP (4)
    • 대외활동 (24)
      • 우아한테크코스 (14)
      • DND 동아리 (4)
      • UMC 동아리 (5)
      • 해커톤 (1)
    • Android (29)
      • MVVM (2)
      • 스터디 (11)
      • Compose (3)
      • Unit Test (1)
    • Project (5)
      • 어따세워 (5)
      • DnD 과외 서비스 (0)
    • Programming (11)
      • Kotlin (4)
      • 파이썬 (7)
    • Git (1)
    • 인공지능 (22)
    • 백준 (8)
    • 기타 (3)
      • IntelliJ (1)
      • 일상 (0)

블로그 메뉴

  • 홈

공지사항

인기 글

태그

  • 원시값 포장
  • RecyclerView
  • 인공지능 분류
  • Baekjoon
  • ViewModel
  • 외부 단편화
  • 백준
  • 우테코 프리코스
  • 컴공선배
  • 우아한테크코스
  • 선형회귀
  • Compose
  • k-means++
  • 인공지능
  • Ai
  • UMC
  • 우테코
  • 객체지향 생활체조
  • 어따세워
  • 다이나믹 프로그래밍
  • MVVM
  • External fragmentation
  • 안드로이드
  • 우테코 5기
  • Android
  • 셀레니움
  • 운영체제
  • 파이썬
  • K-means
  • 딥러닝

최근 댓글

최근 글

티스토리

hELLO · Designed By 정상우.
BuNa_

IT Story

[Android] Thread의 interrupt()와 stop()
Android/스터디

[Android] Thread의 interrupt()와 stop()

2022. 5. 2. 14:44

 

■ Thread.stop() Deprecated 된 이유

 

Thread는 run() 메서드 내부의 코드가 모두 실행되면 자동적으로 종료됩니다. 하지만, 경우에 따라 개발자는 실행 중인 Thread를 즉시 종료할 필요가 발생할 수 있습니다. 예를 들어, 음악이 틀어져 있는 상태에서 음악의 진행도를 Progress로 표현하고자할 때 Thread를 사용한다면, 음악이 종료되었을 때, 더 이상 Progress를 변화시킬 필요가 없으므로 Thread를 종료시켜 메모리를 절약해야 합니다.

이전에는 이러한 경우에 Thread의 stop() 메서드를 호출하여 종료시켰지만, Oracle이 제공하는 Java API 문서에 따르면, This method is inherently unsafe. 를 시작으로 메서드가 사라졌음을 알려줍니다. 말 그대로, stop() 메서드를 통해 Thread가 갑자기 종료되면, Thread가 사용 중이던 자원들이 Unsafe한 상태로 남겨지기 때문에 데드락 등 여러 문제를 초래할 수 있습니다.

 

■ Thread를 안전하게 종료하는 여러가지 방법들

 

deprecated된 stop() 메서드 대신, Thread를 safe하게 종료할 수 있는 여러가지 방법들이 있습니다.

차근차근 하나씩 살펴보도록 하겠습니다.

 

□ run() 메서드내에 flag를 사용하여 종료

class MyThread(var stopFlag: Boolean = false) : Thread(){

    override fun run() {
        super.run()
        
        while(!stopFlag) {
            Log.d("Thread", "실행중")
        }
        
        Log.d("Thread", "자원 정리")
        Log.d("Thread", "실행 종료")
    }
}

 

위의 코드에서 외부에서 stopFlag가 true로 설정하면, 실행하고 있던 while문의 코드 블럭을 빠져나와, Thread가 사용 중이던 자원을 정리하고, run() 메서드가 끝나게 됨으로써 Thread가 safe하게 종료됩니다.

 

□ Thread의 interrupt() 메서드

interrupt는 운영체제 관점에서 보았을 때, CPU가 입출력을 요청하고 나서, I/O Device에 의해 입출력 완료 신호를 전송받는 하나의 신호입니다. 일반적으로 시분할 (TimeSharing) 시스템이나 멀티 프로그래밍 방식에서 interrupt가 발생하면 다음 Process에게 CPU 사용 권한이 넘어갑니다.

Thread에서 interrupt()는 Thread가 일시 정지 상태에 있을 때, InterruptedException 예외를 발생시키는 역할을 한니다.

여기서 중요한 점은, sleep()과 같은 일시정지 상태가 되는 메서드가 호출이 되어야 한다는 점입니다.

만약 run() 메서드 내에 일시정지되는 부분이 없다면, interrupt() 메서드를 호출한다 하더라도, 본질적으로 예외 처리(catch) 블록으로 이동하지 않기 때문입니다.

 

class MyThread : Thread(){

    override fun run() {
        super.run()

        while(true) {
            try {
                sleep(1L) // 잠깐이라도 일시정지가 되어야 함.
                Log.d("Thread", "실행중")
            }catch (e: InterruptedException) {
            	Log.d("Thread", "인터럽트 발생")
                Log.d("Thread", "자원 정리")
                Log.d("Thread", "실행 종료")
            }
        }

    }
}

fun main() {
    val myThread = MyThread()
    myThread.interrupt() // 인터럽트 발생
}

 

위의 코드에서 정상적으로 Thread 내의 while문을 반복하다가 외부에서 interrupt() 메서드를 호출하면 예외 처리(catch)블록으로 이동하여 Thread가 종료될 것입니다.

만약 sleep(1L)가 없다면, 앞서 말씀드린 것처럼 interrupt() 메서드 호출은 아무런 의미가 없어지게 됩니다.

하지만 sleep을 사용하는 것은 결코 최선의 해결책이라고 보기엔 어렵다고 생각합니다.

실제 프로젝트를 구현함에 있어서 sleep을 통해 잠시 일시정지가 된다는 것은 자원을 비효율적으로 사용한다고 볼 수 있기 때문입니다.

 

□ Thread의 isInterrupted 조건

Thread 클래스에는 getIsInterrupted() 라는 유용한 메서드를 제공합니다. 해당 메서드는 interrupt() 메서드가 호출되면 값이 True로 변하면서 더 이상 Thread가 동작할 필요가 없음을 알려줍니다.

class MyThread : Thread(){

    override fun run() {
        super.run()

        while(true) {
            if(isInterrupted) {
                Log.d("Thread", "인터럽트 발생")
                Log.d("Thread", "자원 정리")
                Log.d("Thread", "실행 종료")
                break
            }
            Log.d("Thread", "실행중")
        }

    }
}

fun main() {
    val myThread = MyThread()
    myThread.interrupt() // 인터럽트 발생
}

 

이전 코드와 달라진 점이 있다면, sleep(1L) 메서드가 사라지고 isInterrupted라는 조건문이 하나 추가되었으며, 예외 처리 (try ... catch) 블럭이 사라졌다는 것입니다. 이전에 비해 훨씬 코드가 깔끔해지고, 일시정지 상태가 되지 않기 때문에,  sleep을 통해 잠시 시간이 지체되는 것보다 비교적 Thread를 빠르게 종료시킬 수 있습니다.

 

 

저작자표시 비영리 변경금지 (새창열림)

'Android > 스터디' 카테고리의 다른 글

[Android] ConstraintLayout을 사용해야 하는 이유  (1) 2023.02.22
[Android] ViewTreeObserver를 이용하여 View가 그려지는 시점을 알아보자! (getWidth(), getHeight()의 출력값이 0인 경우..)  (0) 2022.02.19
쿠키(Cookie)와 세션(Session)을 이용한 로그인  (2) 2022.01.21
[Android] 안드로이드 Strings.xml 국가별 언어 설정 방법 및 국가 코드 정리  (5) 2021.12.28
[Android] RecyclerView LayoutPosition vs AdapterPosition 차이를 알아보자!  (0) 2021.12.22
    'Android/스터디' 카테고리의 다른 글
    • [Android] ConstraintLayout을 사용해야 하는 이유
    • [Android] ViewTreeObserver를 이용하여 View가 그려지는 시점을 알아보자! (getWidth(), getHeight()의 출력값이 0인 경우..)
    • 쿠키(Cookie)와 세션(Session)을 이용한 로그인
    • [Android] 안드로이드 Strings.xml 국가별 언어 설정 방법 및 국가 코드 정리
    BuNa_
    BuNa_
    안드로이드 개발자를 향해 달리고 있는 공대생입니다! 🧑 Android, Kotlin, Java, Python 등 학습하고 있는 내용과 프로젝트를 주로 업로드하고 있습니다. 지적과 조언은 언제나 환영입니다!😊 github : https://github.com/tmdgh1592

    티스토리툴바