[Java] DaemonThread(守护线程)简介---finally内的代码不一定会被执行

来源:互联网 发布:memcache 教程php 编辑:程序博客网 时间:2024/06/05 19:07
原文:http://www.cnblogs.com/luochengor/archive/2011/08/11/2134818.html
对其示例代码有所改动。
另最后增加两个工具的笔记。

在Java中有两类线程:用户线程 (User Thread)、守护线程 (Daemon Thread)。

所谓守护 线程,是指在程序运行的时候在后台提供一种通用服务的线程,比如垃圾回收线程就是一个很称职的守护者,并且这种线程并不属于程序中不可或缺的部分。因 此,当所有的非守护线程结束时,程序也就终止了,同时会杀死进程中的所有守护线程。反过来说,只要任何非守护线程还在运行,程序就不会终止。

用户线程和守护线程两者几乎没有区别,唯一的不同之处就在于虚拟机的离开:如果用户线程已经全部退出运行了,只剩下守护线程存在了,虚拟机也就退出了。 因为没有了被守护者,守护线程也就没有工作可做了,也就没有继续运行程序的必要了。

将线程转换为守护线程可以通过调用Thread对象的setDaemon(true)方法来实现。在使用守护线程时需要注意一下几点:

(1) thread.setDaemon(true)必须在thread.start()之前设置,否则会跑出一个IllegalThreadStateException异常。
你不能把正在运行的常规线程设置为守护线程。
(2) 在Daemon线程中产生的新线程也是Daemon的。
(3) 守护线程应该永远不去访问固有资源,如文件、数据库,因为它会在任何时候甚至在一个操作的中间发生中断。


示例代码:
package lab.sodino.deamon;/** * Finally shoud be always run ? */public class DaemonsDontRunFinally {    public static void main(String[] args) {    System.out.println("Daemon Thread test..");        Thread t = new Thread(new RDaemon());        t.setDaemon(true);        t.start();    }}package lab.sodino.deamon;import java.util.concurrent.TimeUnit;public class RDaemon implements Runnable {public void run() {boolean isDaemon = Thread.currentThread().isDaemon();try {StringBuffer sb = new StringBuffer();sb.append("start Thread, daemon:" + isDaemon);if (isDaemon) {sb.append(", 'finally does not run!'");} else {sb.append(", 'finally does run!'");}System.out.println(sb.toString());TimeUnit.SECONDS.sleep(1);} catch (Exception e) {if (e instanceof InterruptedException) {System.out.println("Exiting via InterruptedException");}} finally {System.out.println("Finally has run.");}}}



运行结果:
Daemon Thread test..start Thread, daemon:true, 'finally does not run!'

如果将main函数中的t.setDaemon(true);注释掉,运行结果如下:
Daemon Thread test..start Thread, daemon:false, 'finally does run!'Finally has run.



可以使用jvisualVM.exe监测java进程中的线程详情。


或jstack [pid]命令亦可



0 0
原创粉丝点击