线程分类:java
User Thread(用户线程)spa
Daemon Thread(守护线程) .net
定义:线程
守护线程--也称“服务线程”,在没有用户线程可服务时会自动离开。
优先级:code
守护线程的优先级比较低,用于为系统中的其它对象和线程提供服务。
设置: 对象
经过setDaemon(true)来设置线程为“守护线程”;将一个用户线程设置为守护线程的方式是在 线程对象建立 以前 用线程对象的setDaemon方法。example: 垃圾回收线程就是一个经典的守护线程,当咱们的程序中再也不有任何运行的Thread,程序就不会再产生垃圾,垃圾回收器也就无事可作,因此当垃圾回收线程是JVM上仅剩的线程时,垃圾回收线程会自动离开。它始终在低级别的状态中运行,用于实时监控和管理系统中的可回收资源。blog
生命周期: 生命周期
守护进程(Daemon)是运行在后台的一种特殊进程。它独立于控制终端而且周期性地执行某种任务或等待处理某些发生的事件。也就是说守护线程不依赖于终端,可是依赖于系统,与系统“同生共死”。那Java的守护线程是什么样子的呢。当JVM中全部的线程都是守护线程的时候,JVM就能够退出了;若是还有一个或以上的非守护线程则JVM不会退出。进程
用个比较通俗的好比,任何一个守护线程都是整个JVM中全部非守护线程的保姆: 只要当前JVM实例中尚存在任何一个非守护线程没有结束,守护线程就所有工做;只有当最后一个非守护线程结束时,守护线程随着JVM一同结束工做。Daemon的做用是为其余线程的运行提供便利服务,守护线程最典型的应用就是 GC (垃圾回收器),它就是一个很称职的守护者。User和Daemon二者几乎没有区别,惟一的不一样之处就在于虚拟机的离开:若是 User Thread已经所有退出运行了,只剩下Daemon Thread存在了,虚拟机也就退出了。 由于没有了被守护者,Daemon也就没有工做可作了也就没有继续运行程序的必要了。值得一提的是,守护线程并不是只有虚拟机内部提供,用户在编写程序时也能够本身设置守护线程。下面的方法就是用来设置守护线程的。事件
Thread daemonTread = new Thread(); // 设定 daemonThread 为 守护线程,default false(非守护线程) daemonThread.setDaemon(true); // 验证当前线程是否为守护线程,返回 true 则为守护线程 daemonThread.isDaemon();
守护线程须要注意点:
(1) thread.setDaemon(true)必须在thread.start()以前设置,不然会跑出一个IllegalThreadStateException异常。你不能把正在运行的常规线程设置为守护线程。
(2) 在Daemon线程中产生的新线程也是Daemon的。
(3) 不要认为全部的应用均可以分配给Daemon来进行服务,好比读写操做或者计算逻辑。 (不能保证,当用户进程都退出了,守护进程的 读写任务是否完成,即便没有完成,守护进程也会自动退出)
守护进程进行读写文件操做的例子:
//完成文件输出的守护线程任务 import java.io.*; class TestRunnable implements Runnable{ public void run(){ try{ Thread.sleep(1000);//守护线程阻塞1秒后运行 File f=new File("daemon.txt"); FileOutputStream os=new FileOutputStream(f,true); os.write("daemon".getBytes()); } catch(IOException e1){ e1.printStackTrace(); } catch(InterruptedException e2){ e2.printStackTrace(); } } } public class TestDemo2{ public static void main(String[] args) throws InterruptedException { Runnable tr=new TestRunnable(); Thread thread=new Thread(tr); thread.setDaemon(true); //设置守护线程 thread.start(); //开始执行分进程 } } //运行结果:文件daemon.txt中没有"daemon"字符串。
缘由也很简单,直到主线程完成,守护线程仍处于1秒的阻塞状态。这个时候主线程很快就运行完了,虚拟机退出,Daemon中止服务,输出操做天然失败了。
守护进程和用户进程同时执行任务的例子:
public class Test { public static void main(String args) { Thread t1 = new MyCommon(); Thread t2 = new Thread(new MyDaemon()); t2.setDaemon(true); //设置为守护线程 t2.start(); t1.start(); } } class MyCommon extends Thread { public void run() { for (int i = 0; i < 5; i++) { System.out.println("线程1第" + i + "次执行!"); try { Thread.sleep(7); } catch (InterruptedException e) { e.printStackTrace(); } } } } class MyDaemon implements Runnable { public void run() { for (long i = 0; i < 9999999L; i++) { System.out.println("后台线程第" + i + "次执行!"); try { Thread.sleep(7); } catch (InterruptedException e) { e.printStackTrace(); } } } }
执行的结果为:
后台线程第0次执行!
线程1第0次执行!
线程1第1次执行!
后台线程第1次执行!
后台线程第2次执行!
线程1第2次执行!
线程1第3次执行!
后台线程第3次执行!
线程1第4次执行!
后台线程第4次执行!
后台线程第5次执行!
后台线程第6次执行!
后台线程第7次执行!
Process finished with exit code 0
当用户线程运行结束后,守护线程任务并无结束就退出了。
转载自:
http://blog.csdn.net/shimiso/article/details/8964414