前言

今天在開發中對Java程式的退出產生了困惑,因為題主之前寫過一段時間Go,這兩者的程式退出邏輯是不同的,下面首先給出結論,再通過簡單的例子來介紹。

對於Java程式,Main執行緒退出,如果當前存在非守護執行緒,則Java程式會等待非守護執行緒都執行完再退出;如果只存在守護執行緒,則會直接退出。這是JVM底層實現的機制。

對於Go程式,如果main協程已經退出,那麼其他任何協程都將退出。在非main協程中建立的子協程,如果父協程退出了,子協程依然可以正常執行。

Java程式退出

package main.java.io;

import java.io.IOException;

public class Test {
/**
* main執行緒退出後,如果當前只存在其他的守護執行緒,則程式會直接退出;
* 如果存在非守護執行緒,則會等待其他守護執行緒執行完畢,這是jvm底層實現機制
*/
public static void main(String[] args) throws IOException {
System.out.println("main start");
Thread t1 = new Thread(() -> {
System.out.println("t1 start");
try {
Thread.sleep(2000);
} catch (InterruptedException e) {
e.printStackTrace();
}
System.out.println("t1 exit");
});
t1.setDaemon(true);
t1.start();
System.out.println("main exit");
}
}

如果沒有t1.setDaemon(true),那麼主執行緒退出後,t1執行緒執行完之後程式才退出;否則,主程式退出後程序直接終止。

Go程式退出

package main

import (
"fmt"
"time"
) func test() {
go func() {
fmt.Println("father start")
go func() {
fmt.Println("son start")
time.Sleep(time.Second)
fmt.Println("son exit")
}()
fmt.Println("father exit")
}()
} func main() {
fmt.Println("main start")
test()
time.Sleep(time.Second * 2)
go func() {
fmt.Println("t1 start")
fmt.Println("t1 exit")
}()
fmt.Println("main exit")
} // 結果:
// main start
// father start
// father exit
// son start
// son exit
// main exit

father協程已經退出,但son協程依然執行了。當main協程退出後,t1協程也直接終止。

參考