前言
今天在開發中對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
協程也直接終止。