Kotlin
[Kotlin] CoroutineContext
꾸준함.
2024. 10. 3. 12:48
CoroutineContext
- 코틀린에서 코루틴을 실행할 때 사용하는 컨텍스트를 나타내는 인터페이스
- Spring MVC와 같이 thread-per-request 모델에서는 ThreadLocal을 사용하지만 Reactor나 코루틴처럼 각 요청을 처리할 때마다 수행하는 쓰레드가 변경되기 때문에 특정 쓰레드에 종속되는 ThreadLocal에 접근 불가
- 이에 따라 Reactor에서는 Context, 코루틴에서는 CoroutineContext를 사용하여 파이프라인 context를 저장
- CoroutineContext는 코루틴의 동작을 커스터마이징하고, 실행 쓰레드나 예외 처리, 디버깅 정보 등을 포함하는 요소들을 담고 있음
1. Continuation
- suspend 함수는 다양한 쓰레드에서 실행되기 때문에 ThreadLocal 사용 불가
- 앞선 게시글에서 설명한 Continuation을 통해 실행 상태와 관련된 데이터를 전달
- Continuation은 CoroutineContext를 포함하고 해당 코루틴의 실행 환경과 필요한 정보를 지속적으로 전달
- 코루틴이 suspend 함수 내에서 중단되고 다시 재개될 때, 해당 코루틴의 전체 상태를 이어받아야 하므로, CoroutineContext는 해당 정보를 계속 유지하고 있어야 함
2. CoroutineContext 접근 방법
- runBlocking, launch, aysnc와 같은 CoroutineScope 내부에서는 CoroutineScope.coroutineContext를 통해 접근 가능
- Continuation에 접근 가능할 경우 Continuation.coroutineContext를 통해 접근 가능
- suspend 함수 내부에서는 coroutineContext를 통해 접근 가능
2.1 예시 코드
부연 설명
- runBlocking CoroutineScope 내부에서 coroutineContext 접근
- suspend 함수 내부에서 coroutineContext 접근
- suspendCoroutine 내부에서 continuation을 인자로 받고 continuation.context로 접근
3. CoroutineContext 객체
- Key는 Element를 구분할 때 사용하는 식별자
- CoroutineContext는 여러 Element를 포함하며 Element의 개수에 따라 다른 객체로 존재
- EmptyCoroutineContext: Element가 하나도 없는 상태
- Element: Element가 하나인 상태로 Element 그 자체
- CombinedContext: Element가 두 개 이상일 때
4. CoroutineContext 전파
4.1 suspend 함수 간 전파
- suspend 함수에서 다른 suspend 함수를 호출하는 경우 외부 suspend 함수의 Continuation를 전달하며 이를 통해 외부 Continuation의 CoroutineContext가 내부 suspend 함수에 전달
4.2 withContext
- 현재 context에 특정 Element만 추가해서 실행하고 싶을 경우 사용
- 현재 코루틴의 coroutineContext에 인자로 전달된 context를 merge
- 새로운 Job을 생성해 주입
부연 설명
- runBlocking 내부에서 withContext 실행
- withContext는 runBlocking의 coroutineContext를 merge 하여 CoroutineName을 오버라이드하고 UndispatchedCoroutineJob을 새로 생성
5. CoroutineContext 종류
5.1 CoroutineName
- 디버깅에 이용되는 Element
5.2 Job
- 코루틴의 생명주기 관리
- Job은 active, completed, cancelled와 같은 여러 상태를 갖음
- start, cancel을 통해 명시적으로 시작과 취소 가능
- parent, children을 통해 다른 코루틴의 생명주기도 관리
- launch, async 등의 코루틴 비리더를 통해 자식 Job 생성 가능
https://kotlinlang.org/api/kotlinx.coroutines/kotlinx-coroutines-core/kotlinx.coroutines/-job/
5.3 ReactorContext
- ReactorContext를 통해 Reactor의 ContextView를 다른 suspend 함수에 전달
부연 설명
- contextWrite을 통해 context 주입
- launch 블록 내 coroutineContext의 ReactorContext로 접근하여 해당 context를 기반으로 새로운 reactorContext 생성
- ReactorContext(newContext)로 launch에 전달
- Mono에서 contextWrite을 통해 coroutineContext로 전달된 reactorContext를 주입하고 출력
6. 코루틴 예외 처리
6.1 aysnc의 예외 처리
- CoroutineScope aysnc 내에서 예외가 발생할 경우 동기 코드처럼 try-catch를 통해 예외 처리 가능
6.2 launch의 예외 처리
- CoroutineScope launch 내에서 예외가 발생하는 경우 async와 달리 try-catch 블록에 잡히지 않고 쓰레드의 UncaughtExceptionHandler를 통해 출력
- launch 전체를 try-catch로 감싸더라도 UncaughtExceptionHandler를 통해 출력
- 함수처럼 exception이 전파되는 구조가 아니라 자식 Job에서 부모 Job으로 cancellation이 전파되며 함께 exception이 전달되기 때문에 일반적인 try-catch로 예외처리 불가
6.3 CoroutineExceptionHandler
- CoroutineExceptionHandler를 통해 root coroutine의 예외 처리 가능
- launch에 적용 가능
- async는 예외를 Deferred를 통해 전달하기 때문에 적용 불가
- runBlocking에도 적용 불가
- handler는 반드시 root 코루틴 launch에 제공해야 함
- 중간 launch에 제공할 경우 예외 처리 불가
7. CoroutineDispatcher
- 코루틴이 어느 쓰레드에서 실행될지 결정하는 Element
- CoroutineDispatcher는 Default, Main, Unconfined, IO 등을 미리 만들어 제공
- 하지만 Main Dispatcher는 라이브러리를 통해서 주입받아야 하는데 kotlinx-coroutines-core에서 지원하지 않기 때문에 Main은 사용 불가
- Default: 기본으로 사용되는 Dispatcher로 CPU 코어 수만큼 고정된 크기를 갖는 쓰레드 풀을 제공하며 CPU Bound Blocking에 적합
- IO: 기본적으로 최대 64개까지 늘어나는 가변 크기를 갖는 쓰레드 풀을 제공하며 I/O Bound Blocking에 적합 (Reactor의 boundedElastic 쓰레드 풀과 비슷)
- Unconfined: 처음엔 caller 쓰레드 기준으로 실행되고 이후엔 마지막으로 실행된 suspend 함수의 쓰레드를 따라가기 때문에 어떤 쓰레드에서 코루틴이 실행될지 예상하기 힘듦 (일반적으로 사용 X)
부연 설명
- runBlocking은 BlockingEventLoop.dispatcher를 사용
- CoroutineScope를 만들고 Dispatcher를 전달하지 않아 Default로 실행
- Default와 IO Dispatcher는 DefaultDispatcher-worker-1에서 동일하게 동작
- IO와 Default는 쓰레드 풀을 공유하지만 동시에 수행 가능한 쓰레드 수는 Default는 고정인 반면 IO는 가변
- CPU Bound Blocking 작업은 CPU 숫자보다 늘려도 의미가 없는 반면 IO Bound Blocking 작업은 쓰레드를 늘리면 그만큼 더 많은 IO 수행 가능
8. ExecutorCoroutineDispatcher
- 특정한 쓰레드 혹은 고정된 개수의 쓰레드 풀을 갖는 Dispatcher 생성
- N개의 쓰레드를 갖는 Executor를 생성하고 asCoroutineDispatcher를 통해 CoroutineDispatcher로 변환
- ExecutorCoroutineDispatcher는 내부에 Executor를 포함시키기 때문에 명시적으로 close를 호출하여 Executor를 종료해야 함
부연 설명
- runBlocking은 main 쓰레드에서 실행
- withContext는 Dispatchers.IO로 실행
- delay는 DefaultExecutor로 실행
참고
패스트 캠퍼스 - Spring Webflux 완전 정복 : 코루틴부터 리액티브 MSA 프로젝트까지
반응형