提问者:小点点

主线程退出时协程会发生什么?


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

如您所见,当主线程退出时,协程中的代码不再运行。似乎静态编程语言终止了引擎盖下的所有协程。

有人能告诉我主线程退出时协程到底会发生什么吗?


共2个答案

匿名用户

协程本身并不以JVM知道的方式“运行”。它们只不过是堆上的对象。

但是,协程上下文对何时允许终止JVM有发言权。创建您自己的:

val threadPool = Executors.newFixedThreadPool(4)
val dispatcher = threadPool.asCoroutineDispatcher()

现在,如果您使用它而不是Common Pool

launch(dispatcher) { ... }

你会发现即使所有任务都完成了,JVM也不会消失。只有当你明确地说

threadPool.shutdown()

但是请注意,threadPool. Shutdown()对协程的行为与对您提交给它的“经典”任务的行为不同。执行器将确保所有提交的任务在关闭之前完成,但它没有对挂起的协程的帐户。

匿名用户

当主线程执行完毕,进程/JVM实例死亡时,协程终止,它们就像守护线程。参考官方协程指南中的这一节。