Java当主线程退出时,所有用户线程(非守护线程)将继续运行,直到它们完成工作。
我有一个简单的程序,可以将1到5的计数器打印到控制台。
Java版本:
fun main(args: Array<String>) {
println("Main start")
countWithThread()
println("Main end")
}
fun countWithThread() {
Thread(Runnable {
for (i in 1..5) {
println("${Thread.currentThread().name} count $i")
Thread.sleep(10)
}
}).start()
}
输出:
Main start
Main end
Thread-0 count 1
Thread-0 count 2
Thread-0 count 3
Thread-0 count 4
Thread-0 count 5
Process finished with exit code 0
静态编程语言版本:
fun main(args: Array<String>) {
println("Main start")
countWithCoroutine()
println("Main end")
}
fun countWithCoroutine() {
launch(CommonPool) {
for (i in 1..5) {
println("${Thread.currentThread().name} count $i")
delay(10)
}
}
}
输出:
Main start
Main end
Process finished with exit code 0
如您所见,当主线程退出时,协程中的代码不再运行。似乎静态编程语言终止了引擎盖下的所有协程。
有人能告诉我主线程退出时协程到底会发生什么吗?
协程本身并不以JVM知道的方式“运行”。它们只不过是堆上的对象。
但是,协程上下文对何时允许终止JVM有发言权。创建您自己的:
val threadPool = Executors.newFixedThreadPool(4)
val dispatcher = threadPool.asCoroutineDispatcher()
现在,如果您使用它而不是Common Pool
:
launch(dispatcher) { ... }
你会发现即使所有任务都完成了,JVM也不会消失。只有当你明确地说
threadPool.shutdown()
但是请注意,threadPool. Shutdown()
对协程的行为与对您提交给它的“经典”任务的行为不同。执行器将确保所有提交的任务在关闭之前完成,但它没有对挂起的协程的帐户。
当主线程执行完毕,进程/JVM实例死亡时,协程终止,它们就像守护线程。参考官方协程指南中的这一节。