java中的有两种线程:守护线程(Daemon)和用户线程(User)。
守护线程是一直在JVM后台中运行着的线程,为其他线程(用户线程)服务的。垃圾回收线程GC线程就是守护线程。
守护线程随着程序在JVM中运行,守护线程会在第一时间被启动,并且一直处于运行态。只有当所有的用户线程都执行完毕后,程序就会杀死守护线程,然后离开JVM,终止程序。
任何线程都可以设置为守护线程和用户线程,并且可以通过方法Thread.setDaemon(bool on)当bool值为true时则是将用户线程设置为守护线程,成为当前程序的守护线程之一运行到程序退出。当Thread.setDaemon(bool on)的bool值为false则为用户线程。
另外Thread.setDaemon()必须在Thread.start()之前调用,不能将正在运行的线程设置为守护线程,否则运行时会抛出异常;还有在守护线程中产生的新线程也是守护线程; 守护线程不能用来进行JVM中文件、数据库的读写或者进行计算任务,因为当所以用户线程完成后程序会强行杀死所有的守护线程,如果在守护线程中进行读写或计算的话有可能因为程序的退出而导致操作不能完成。
通过Thread Dump打印出来的线程信息,含有Daemon的线程就是守护进程,一般会有:编译守护进程、 服务守护进程、Finalizer守护进程、引用处理守护进程以及经典的GC守护进程。
守护线程和用户线程两者的唯一的区别之处:
在于判JVM何时离开,守护线程是为其他线程提供服务,所以如果所有的用户线程都已经执行完毕,那么守护线程就没有了可以服务的线程,这时JVM撤离。也可以理解为守护线程是JVM自动创建的线程(但不一定),用户线程是程序创建的线程;比如JVM的垃圾回收线程是一个守护线程,当所有线程已经撤离,不再产生垃圾,守护线程自然就没事可干了,当垃圾回收线程是Java虚拟机上仅剩的线程时,Java虚拟机会自动离开。
public class ThreadDemo { public static void main(String[] args) { new Thread(new Runnable() { public void run() { Thread.sleep(3000); } catch (InterruptedException e) { e.printStackTrace(); System.out.println("线程thread结束"); System.out.println("主线程结束"); 复制代码
先输出主线程结束,过3秒输出线程thread结束。
public class ThreadDemo1 { public static void main(String[] args) { Thread thread = new Thread(new Runnable() { public void run() { Thread.sleep(3000); } catch (InterruptedException e) { e.printStackTrace(); System.out.println("线程thread 结束运行"); //在调用start方法之前将tread设置为守护线程 thread.setDaemon(true); System.out.println("main线程运行结束"); 复制代码
只输出mian线程运行结束,不会输出线程thread 结束运行。
因为主线程一旦运行结束,则jvm直接退出,不管守护线程是否运行结束都要退出。
总结:
java中的线程分为两种: 守护线程(Daemon)和用户线程(User)。任何线程都可以设置为守护线程和用户线程,通过方法Thread.setDaemon(bool on);true则把该线程设置为守护线程,反之则为用户线程。Thread.setDaemon()必须在Thread.start()之前调用,否则运行时会抛出异常。
守护线程:指在程序运行的时候在后台提供一种通用服务的线程,比如垃圾回收线程就是一个很称职的守护者,并且这种线程并不属于程序中不可或缺的部分。因 此,当所有的非守护线程结束时,程序也就终止了,同时会杀死进程中的所有守护线程。反过来说,只要任何非守护线程还在运行,程序就不会终止。
用户线程:自己程序或者系统中创建的线程。比如:new Thread。这就是自己创建了一个线程。