守护线程

一、什么是守护线程

在Java中有两类线程:User Thread(用户线程)、Daemon Thread(守护线程)
用个比较通俗的比如,任何一个守护线程都是整个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来进行服务,比如读写操作或者计算逻辑。 因为你不可能知道在所有的User完成之前,Daemon是否已经完成了预期的服务任务。一旦User退出了,可能大量数据还没有来得及读入或写出,计算任务也可能多次运行结果不一样。这对程序是毁灭性的。造成这个结果理由已经说过了:一旦所有User Thread离开了,虚拟机也就退出运行了。
//完成文件输出的守护线程任务  
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   

从上面的执行结果可以看出:

  • 前台线程是保证执行完毕的,后台线程还没有执行完毕就退出了。
  • 实际上:JRE判断程序是否执行结束的标准是所有的前台执线程行完毕了,而不管后台线程的状态,因此,在使用后台线程时候一定要注意这个问题。

Java中守护线程的总结

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值