守护线程以及要使用时注意的一点(Daemon Thread)

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

Daemon的作用是为其他线程的运行提供便利服务,比如垃圾回收线程就是一个很称职的守护者。User和Daemon两者几乎没有区别,唯一的不同之处就在于虚拟机的离开:如果 User Thread已经全部退出运行了,只剩下Daemon Thread存在了,虚拟机也就退出了。 因为没有了被守护者,Daemon也就没有工作可做了,也就没有继续运行程序的必要了。

值得一提的是,守护线程并非只有虚拟机内部提供,用户在编写程序时也可以自己设置守护线程。下面的方法就是用来设置守护线程的。

public final void setDaemon(boolean on)

这里有几点需要注意:

(1) thread.setDaemon(true)必须在thread.start()之前设置,否则会跑出一个IllegalThreadStateException异常。你不能把正在运行的常规线程设置为守护线程。
(2) 在Daemon线程中产生的新线程也是Daemon的。
(3) 不要认为所有的应用都可以分配给Daemon来进行服务,比如读写操作或者计算逻辑。

因为你不可能知道在所有的User完成之前,Daemon是否已经完成了预期的服务任务。一旦User退出了,可能大量数据还没有来得及读入或写出,计算任务也可能多次运行结果不一样。这对程序是毁灭性的。造成这个结果理由已经说过了:一旦所有User Thread离开了,虚拟机也就退出运行了。

 

import java.io.File;
import java.io.FileOutputStream;
import java.io.IOException;
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(); //开始执行分进程     

        System.out.print("success");
    }     
}
class TestRunnable implements Runnable{     
    public void run(){     
        try{     
            File f=new File("daemon.txt");     
            Thread.sleep(1000);//守护线程阻塞1秒后运行     这个时候由于主线程在print("success")后就执行完了,
            //因此整个程序只剩下一个守护线程,虚拟机会关闭,从而守护线程还在sleep中就被迫中断,文件也不能写入了
            FileOutputStream os=new FileOutputStream(f,true);     
            os.write("daemon".getBytes());
            os.close();
        }     
        catch(IOException e1){     
            e1.printStackTrace();     
        }     
        catch(InterruptedException e2){     
            e2.printStackTrace();     
        }     
    }     
}  

 

posted @ 2015-10-19 22:29  湘大小生  阅读(945)  评论(0编辑  收藏  举报