Coroutine Deep Dive - 1
Purpose
이 게시글은 일단 코루틴을 사용하는데 좀 더 이해도를 높이기 위해 작성된 글이다. 대부분의 내용은 코루틴 라이브러리를 보고 작성되었으니, 어느정도 신뢰하여도 좋다.
What is Coroutine?
코루틴은 일시중단이 가능한 연산의 인스턴스 이다. 여기서 인스턴스라는 말이 중요한데, 일단 이해하지 말고, 왜 인스턴스라고 하지? 라는 고민을 가지고 이 글을 읽어줬으면 한다. 밑에서 다 설명할 것이다. 일시 중단 된 지점으로 부터 후에 다시 재 실행될 수 있으며, 이때 특정 Thread 에 Bound 되어 있는 것이 아닌, 일시중단을한 Thread 와 다른 Thread 가 와서 작업을 재개할 수도 있다.
Coroutine use cases
Coroutine 개발자들이 설명하는 Coroutine 을 사용하기 좋은 상황은 아래와 같다.
- Channel-based concurrency
- Actor-based concurrency (Actor Model 을 뜻함.)
- 유저 상호작용이 필요한 백그라운드 프로세스 (모달을 뛰우는 작업과 같은 상황)
- 각 Actore 를 State Machine 이 아닌 Sequence 로 구현할때 (요건 잘 모르겠다..)
- Web application flows
suspend function
suspend function 이란 함수앞에 suspend modifier 가 붙어있는 함수를 뜻한다. 현재 Thread 를 Blocking 하지 않고도 suspend 가 가능함을 의미한다. suspend function 은 일반적인 코드에서 실행시키는 것이 불가능하고, 오직 suspending function 이나 Suspending lambdase 를 통해 실행해야 한다.
suspend fun <T> CompletableFuture<T>.await(): T =
suspendCoroutine<T> { cont: Continuation<T> ->
whenComplete { result, exception ->
if (exception == null) // the future has been completed normally
cont.resume(result)
else // the future has completed with an exception
cont.resumeWithException(exception)
}
}
Suspension Point
코루틴이 실행되는 동안 코루틴의 실행이 중단된 지점을 나타내는 Point, 문법적으로 suspension point 는 suspending function 의 호출을 의미하지만, 실질적인 suspension 은 기본 라이브러리가 suspend 를 호출할때 발생합니다. 쉽게 말하면, suspend function 내의 다른 suspend function 이 호출되는 지점이 보통 Suspension point 이다.
- suspending function 은 yield()
나 await()
같은 함수와 같이 suspend modifier 가 붙여져있는 함수를 뜻한다.
- Intellij 기준으로 저 화살표를 보면 쉽다.
Continuation
suspend 된 코루틴의 중단된 지점에서의 상태를 의미한다. 개념적으로는 suspension point 이후로 실행해야할 나머지 Execution 을 나타내준다.
interface Continuation<in T> {
val context: CoroutineContext
fun resumeWith(result: Result<T>)
}
전에도 얘기한 적이 있는 것 같은데, Continuation 은 Callback + 기타 실행 정보 라고 생각하는것이 편하다. 라고 했는데 이제 어떻게 Callback 을 구성하는지 알아보도록 하자.
/**
* File Read 를 하는 함수
*/
suspend fun a() {
println("File Reading....")
}
fun callback() {
println("File read Done")
}
a
라는 함수는 File 을 읽는 기능을 하는 suspend function 이다. a 라는 함수의 실행이 끝마치면 우리는 Callback 으로 "File read Done" 이라는 메세지를 받아야 한다. 위와 같은 경우를 Continuation 으로 어떻게 표현할 수 있을까? 아래와 같은 코드로 표현할 수 있다.
class ContinuationExample(override val context: CoroutineContext = EmptyCoroutineContext) : Continuation<Unit> {
override fun resumeWith(result: Result<Unit>) {
callback()
}
}
suspend fun main() {
::a.startCoroutine(ContinuationExample())
}
resumeWith
를 통해서 위와 같이 구현이 가능한데, resumeWith 가 completion callback 의 역할을 하기 때문에 위와 같은 형태로 구현이 가능한 것이다. 그렇다면 CoroutineContext 가 어떤것인지 알아보자.
Coroutine Context
Coroutine 을 조금 써봤다면 CoroutineContext 를 만들어 넣어본적이 있을 것이다. 위에서 보여준 EmptyCoroutineContext 도 CoroutineContext 이다. Coroutine Context 는 threading policy, logging, coroutine identity 등의 정보를 저장한다. 쉽게 생각해서 Thread-local 인데 Coroutine 에 Bound 되어 있는 Element Set 이라고 생각하면 된다. CoroutineContext 는 아래와 같은 Interface 를 가지고 있다.
interface CoroutineContext {
operator fun <E : Element> get(key: Key<E>): E?
fun <R> fold(initial: R, operation: (R, Element) -> R): R
operator fun plus(context: CoroutineContext): CoroutineContext
fun minusKey(key: Key<*>): CoroutineContext
interface Element : CoroutineContext {
val key: Key<*>
}
interface Key<E : Element>
}
그래서 우리가 아마 많이 보게되는 CoroutineContext 는 대충 아래와 같은 코드 일 것이다.
launch(context = Dispatchers.IO) {
// do some api call
}
그럼 우리가 만약에 Custom 하게 뭔가 Map 을 넘기거고 싶다거나, 특정 객체를 넘기고 싶은 경우에는 어떻게 사용할 수 있을까? 바로 아래와 같은 방법처럼 AbstractCoroutineContextElement
를 통해서 가능하다. 우리가 어떤 Key 를 사용할 것인지 커스텀하게 만들어서 사용할 수 있다.
class User(
val name: String,
val age: Int
): AbstractCoroutineContextElement(User) {
companion object Key: CoroutineContext.Key<User>
}
suspend fun test() = coroutineScope {
launch(User(name = "roach", age = 10)) {
val userAge = coroutineContext[User]?.age
val userName = coroutineContext[User]?.name
}
}
또한 위의 interface 를 보면 알듯이 기본적으로 plus
와 같은 기능이 존재하여, Threading Policy 가 있는 Context 와 합쳐서 사용할 수 있다.
class User(
val name: String,
val age: Int
): AbstractCoroutineContextElement(User) {
companion object Key: CoroutineContext.Key<User>
}
suspend fun test() = coroutineScope {
launch(User(name = "roach", age = 10) + Dispatchers.IO) {
val userAge = coroutineContext[User]?.age
val userName = coroutineContext[User]?.name
}
}
만약 자신이 Coroutine 에 외부 Scope 의 변수를 Closure 처럼 넘겨서 사용하고 있다면, 위와 같은 방법으로 조금 더 좋게 개선하거나, Indetion 을 증가시키지 않을 수 있는 방법일지를 고민해볼 수 있을 것이다. 사실 Coroutine Context 에 관한 더 깊은 내용은 따로 설명해야 할 정도로 분량이 길기 때문에, 이제 위에서 말한 Coroutine 이 왜 suspendable function 의 Instance 인지 확인해보자.
Coroutine
우리가 아까 작성한 코드를 다시 들고와서 이에 관한 내용을 설명해보려고 한다.
suspend fun a() {
println("File Reading....")
}
suspend fun b() {
println("Read Another file")
}
suspend fun c() {
println("Read Another file")
}
fun callback() {
println("File read Done")
}
class ContinuationExample(override val context: CoroutineContext = EmptyCoroutineContext) : Continuation<Unit> {
override fun resumeWith(result: Result<Unit>) {
callback()
}
}
suspend fun main() {
::a.startCoroutine(ContinuationExample())
b()
c()
}
main function 을 보면 몇개의 suspend function 을 더 추가했다. 위와 같은 코드는 자바 코드로 어떻게 바뀔까?
위와 같이 큰 switch 문으로 변한다. 자세히 보면 switch(label) 을 이용하여 코드를 나눈 것을 확인할 수 있는데, 위에서 설명한 suspension point 를 기반으로 labeling 을 하게 된다. 보면 main function 은 위와 같이 변경되었고, 위의 인스턴스를 실제 Main 문에서 실행하고 있음을 알 수 있다.
// $FF: synthetic method
public static void main(String[] var0) {
RunSuspendKt.runSuspend(new ContinuationExampleKt$$$main(var0));
}
우리의 Style 로 표현
위에서 보여준 코드는 Java 로 Decompile 된 코드라 아마도 이해하기 힘들 것이다. 공식문서에 의거해서 우리가 조금 더 이해할 수 있는 코드로 변경해보자. 현재 main 문의 상태는 아래와 같다.
suspend fun main() {
::a.startCoroutine(ContinuationExample())
b()
c()
}
만약 위의 코드가 바뀌게 되면 어떻게 될까? 우리가 배운 지식들로 조금 더 알아보기 쉽게 바꿔보자.
fun main() {
val mainCoroutine = MainContinuationImpl()
mainCoroutine.resumeWith(Result.success(Unit))
}
class MainContinuationImpl(
var label: Int = 0,
override val context: CoroutineContext = EmptyCoroutineContext
): Continuation<Unit> {
override fun resumeWith(result: Result<Unit>) {
when(label) {
0 -> {
label = 1
a()
callback() // a.resumeWith()
this.resumeWith(Result.success(Unit)) // goto loop
}
1 -> {
label = 2
b()
this.resumeWith(Result.success(Unit)) // goto loop
}
2 -> {
c()
return
}
}
}
}
우리의 Main 문은 위와 같이 변했을 것이다. 이제 조금 이해가 가는가? Main 문은 하나의 StateMachine 으로 변했다. 그래서 우리가 Coroutine 을 suspendable function 의 Instance 라고 하는 이유이다. 여기서 코루틴이 시작하게 되면 첫번째로 우리는 resumeWith 를 처음으로 실행해서 label 0
인 곳으로 goto 하게 된다. label 0
에서의 작업을 마치면 우리는 다음 state 인 label 1
로 가기 위해 label 을 1로 만들어주게 된다.
위의 지식을 좀 더 딥다이브 하기 위해서는 선행적으로 CPS Style에 대해서 조금 알면 좋다. 모르면 아래 내글을 봐도 좋고 아니면 다른 글을 찾아서 봐도 좋다.
https://devroach.tistory.com/149
참고
https://github.com/Kotlin/KEEP/blob/master/proposals/coroutines.md#coroutine-context
Github
https://github.com/tmdgusya/analyze-coroutine
'Kotlin' 카테고리의 다른 글
Coroutine Series-2) Continuation (0) | 2022.09.28 |
---|---|
Coroutine Series-1 ) 코루틴을 왜 사용해야 하는가? (0) | 2022.09.25 |
CPS Style (1) | 2022.07.31 |
Hot And Cold Data Stream (0) | 2022.07.16 |
Kotlin) Channel 을 이용한 Actor Model 구현 (0) | 2022.07.15 |