今天在开发中对Java
程序的退出产生了困惑,因为题主之前写过一段时间Go
,这两者的程序退出逻辑是不同的,下面首先给出结论,再通过简单的例子来介绍。
对于Java
程序,Main线程退出,如果当前存在非守护线程,则Java程序会等待非守护线程都执行完再退出;如果只存在守护线程,则会直接退出。这是JVM底层实现的机制。
对于Go
程序,如果main协程已经退出,那么其他任何协程都将退出。在非main协程中创建的子协程,如果父协程退出了,子协程依然可以正常运行。
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
线程执行完之后程序才退出;否则,主程序退出后程序直接终止。
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
协程也直接终止。