■ 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 |