Kotlin Coroutines vs Java Threads
There are probably many articles about Kotlin’s coroutines online. That’s why I would like to focus just on showing you the difference between concept over coroutines and threads – a well-known concurrency mechanism in Java. We will start with a small portion of the theory.
In general, Kotlin Coroutines are presented as a lightweight alternative to Java Threads. You would probably ask why creating them is much cheaper than creating threads. The answer is very simple – because they are not using such threads as normal threads 😉 Of course, it’s a joke, but the first important thing you should know about coroutines is that they are using thread pools in background. So, it’s not a “magical” technology, that is better than threads, but just a different concept of concurrency used in your applications.
Unlike threads, coroutines are not bound to any particular thread. A coroutine can start executing in one thread, suspend execution, and resume on a different thread. Coroutines are not managed by the operating system, but by the Kotlin Runtime. When you are sleeping a thread it is blocked for a particular period of time. So you can’t use that thread anymore until it finishes its work. In coroutines, we may suspend execution, which means that the current thread is returned to a pool and may be used, for example by another coroutine. Let’s proceed to the examples.
Example
As always a source code with examples is available on GitHub. The address of repository is https://github.com/piomin/sample-kotlin-playground.git. It contains example for another article, so for test of coroutines you should take a look on pl.piomin.services.test.CoroutinesTest
class.
Implementation
Let’s start by creating a simple coroutine. Of course, there are some different ways to create coroutine, but I’m choosing the simplest one – with GlobalScope
. It means that the lifetime of each coroutine is limited only by the lifetime of the whole application or a test as in our case. We are creating coroutine 10 times and after launching it we are printing the name of the currently used thread. We are not calling any suspend method here – I just want to show you how it works.
@Test
fun testSimpleCoroutine() {
var i: Int = 0
repeat(10) {
GlobalScope.launch {
println("${++i}: ${Thread.currentThread().name}")
}
}
Thread.sleep(100)
}
The order of printing messages is indeterminate. However, you may see that it was using just 3 different thread during test execution. A default thread pool used here is called DefaultDispatcher-worker
. It also add the name of coroutine to the thread name.
We may compare it to the result of this test. It does the same thing as the previous test, but uses Java thread instead of Kotlin coroutine.
@Test
fun testSimpleThread() {
var j: Int = 0
repeat(10) {
Thread(Runnable {
println("${++j}: ${Thread.currentThread().name}")
}).start()
}
Thread.sleep(100)
}
Here’s the result. As you probably expect the order of printing messages is still indeterminate (even more than earlier, but we will discuss it in the next section), but of course there were 10 running threads during the test. In comparison to the previous test it had to create 10 threads instead of 3.
It is said that Kotlin coroutines are processed sequentially. What does it mean in practice? Let’s change the default thread pool used by our coroutines to force them using just a main thread. To do that we need to override default CoroutineDispatcher
inside launch
method with Dispatchers.Unconfined
.
@Test
fun testSimpleCoroutineThreadMain() {
var i: Int = 0
repeat(10) {
GlobalScope.launch(Dispatchers.Unconfined) {
i++
println("$i: ${Thread.currentThread().name}")
}
}
Thread.sleep(100)
}
No matter how many times you will run that test, the result will be always the same. The messages are printing in order of launching new coroutines.
And finally we may proceed to the last test. We will call suspension method delay
inside our coroutine. We will also print the name of currently used thread before and after calling delay
.
@Test
fun testSimpleCoroutineWithDelay() {
repeat(10) {
GlobalScope.launch {
println("Before delay $it: ${Thread.currentThread().name}")
delay(10)
println("After delay $it: ${Thread.currentThread().name}")
}
}
Thread.sleep(200)
}
You can easily verify that after suspension the rest of job inside coroutine has been processed in different thread than before.
Summary
In this short article I was trying to explain, using simple words, what exactly is Kotlin coroutine. I hope it helps you to understand the most important differences between threads and Kotlin coroutines.
2 COMMENTS