原因: main线程是由java虚拟机在启动的时候创建的。main方法开始执行的时候,主线程已经创建好并在运行了。对于运行中的线程,调用Thread.setDaemon()会抛出异常Exception in thread “main” java.lang.IllegalThreadStateException。测试代码如下:
public class MainTest { public static void main(String[] args) { System.out.println(" parent thread begin "); Thread.currentThread().setDaemon(true); } }主线程,只是个普通的非守护线程,用来启动应用程序,不能设置成守护线程;除此之外,它跟其他非守护线程没有什么不同。主线程执行结束,其他线程一样可以正常执行。代码如下:
public class ParentTest { public static void main(String[] args) { System.out.println("parent thread begin "); ChildThread t1 = new ChildThread("thread1"); ChildThread t2 = new ChildThread("thread2"); t1.start(); t2.start(); System.out.println("parent thread over "); } } class ChildThread extends Thread { private String name = null; public ChildThread(String name) { this.name = name; } @Override public void run() { System.out.println(this.name + "--child thead begin"); try { Thread.sleep(500); } catch (InterruptedException e) { System.out.println(e); } System.out.println(this.name + "--child thead over"); } }程序运行结果如下: parent thread begin parent thread over thread2–child thead begin thread1–child thead begin thread2–child thead over thread1–child thead over
这样其实是很合理的,按照操作系统的理论,进程是资源分配的基本单位,线程是CPU调度的基本单位。对于CPU来说,其实并不存在java的主线程和子线程之分,都只是个普通的线程。进程的资源是线程共享的,只要进程还在,线程就可以正常执行,换句话说线程是强依赖于进程的。也就是说,线程其实并不存在互相依赖的关系,一个线程的死亡从理论上来说,不会对其他线程有什么影响。
java虚拟机(相当于进程)退出的时机是:虚拟机中所有存活的线程都是守护线程。只要还有存活的非守护线程虚拟机就不会退出,而是等待非守护线程执行完毕;反之,如果虚拟机中的线程都是守护线程,那么不管这些线程的死活java虚拟机都会退出。测试代码如下:
public class ParentTest { public static void main(String[] args) { System.out.println("parent thread begin "); ChildThread t1 = new ChildThread("thread1"); ChildThread t2 = new ChildThread("thread2"); t1.setDaemon(true); t2.setDaemon(true); t1.start(); t2.start(); System.out.println("parent thread over "); } } class ChildThread extends Thread { private String name = null; public ChildThread(String name) { this.name = name; } @Override public void run() { System.out.println(this.name + "--child thead begin"); try { Thread.sleep(500); } catch (InterruptedException e) { System.out.println(e); } System.out.println(this.name + "--child thead over"); } }执行结果如下: parent thread begin parent thread over thread1–child thead begin thread2–child thead begin
在这种情况下,的确主线程退出,子线程就立刻结束了,但是这是属于JVM的底层实现机制,并不是说主线程和子线程之间存在依赖关系
转自: https://www.cnblogs.com/qiumingcheng/p/8202393.html