【Java 多线程 5】守护线程

桃扇骨 2021-08-24 20:55 587阅读 0赞

一、守护线程简介

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

1、Daemon的作用

Daemon的作用是为其他线程的运行提供便利服务,守护线程最典型的应用就是 GC (垃圾回收器),它就是一个很称职的守护者。

2、User和Daemon的区别

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

3、用户在编写程序时可以自己设置守护线程

  1. Thread daemonTread = new Thread();
  2. // 设定 daemonThread 为 守护线程,default false(非守护线程)
  3. daemonThread.setDaemon(true);
  4. // 验证当前线程是否为守护线程,返回 true 则为守护线程
  5. daemonThread.isDaemon();

注:

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

4、代码实例

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

  1. import java.io.File;
  2. import java.io.FileOutputStream;
  3. import java.io.IOException;
  4. public class TestRunnable implements Runnable {
  5. public void run(){
  6. try {
  7. Thread.sleep(1000);
  8. File f = new File("daemon.txt");
  9. FileOutputStream os = new FileOutputStream(f,true);
  10. os.write("daemon".getBytes());
  11. }catch (IOException e1){
  12. e1.printStackTrace();
  13. }catch (InterruptedException e2){
  14. e2.printStackTrace();
  15. }
  16. }
  17. public static void main(String[] args) {
  18. Runnable tr = new TestRunnable();
  19. Thread thread = new Thread(tr);
  20. thread.setDaemon(true);
  21. thread.start();
  22. }
  23. }
  24. //运行结果:文件daemon.txt中没有"daemon"字符串。

但是如果设置成普通线程,daemon会写入daemon.txt文件中

13ae51d73c22fba8ca28658f9db7c999d2b.jpg

把输入输出逻辑包装进守护线程,字符串并没有写入指定文件

5、原因

直到主线程完成,守护线程仍处于1秒的阻塞状态,主线程运行完成,虚拟机退出,daemon停止服务,守护线程停止运行。

代码实例:

  1. package reflect;
  2. public class Test {
  3. public static void main(String[] args) {
  4. Thread t1 = new MyCommon();
  5. Thread t2 = new Thread(new MyDaemon());
  6. t2.setDaemon(true);
  7. t2.start();
  8. t1.start();
  9. }
  10. }
  11. package reflect;
  12. public class MyDaemon implements Runnable{
  13. public void run(){
  14. for (int i = 0; i < 1000; i++) {
  15. System.out.println("后台线程第"+i+"次执行!");
  16. try {
  17. Thread.sleep(7);
  18. } catch (InterruptedException e) {
  19. e.printStackTrace();
  20. }
  21. }
  22. }
  23. }
  24. package reflect;
  25. public class MyCommon extends Thread{
  26. public void run(){
  27. for (int i = 0; i < 5; i++) {
  28. System.out.println("线程1第"+i+"次执行!");
  29. try {
  30. Thread.sleep(7);
  31. } catch (InterruptedException e) {
  32. e.printStackTrace();
  33. }
  34. }
  35. }
  36. }

运行结果:

03e1cbbf802c636edffa2c3749ffe6f9356.jpg

从上面的执行结果可以看出:
前台线程是保证执行完毕的,后台线程还没有执行完毕就退出了。

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

二、补充说明

1、定义

守护线程也称服务线程,在没有用户线程可服务时会自动离开。

2、优先级

守护线程的优先级比较低,用于为系统中的其它对象和线程提供服务。

3、通过setDaemon(true)来设置守护线程。

4、example

垃圾回收线程就是一个经典的守护线程,当我们的程序中不再有任何运行的
Thread,程序就不会再产生垃圾,垃圾回收器也就无事可做,所以当垃圾回收线程是
JVM上仅剩的线程时,垃圾回收线程会自动离开。它始终在低级别的状态中运行,用于
实时监控和管理系统中的可回收资源。

5、生命周期

守护进程(Daemon)是运行在后台的一种特殊进程。它独立于控制终端并且
周期性地执行某种任务或等待处理某些发生的事件。也就是
说守护线程不依赖于终端,但是依赖于系统,与系统“同生共死”。那Java的守护线程是
什么样子的呢。当JVM中所有的线程都是守护线程的时候,JVM就可以退出了;如果还有一个或以上的非守护线程则JVM不会退出。

三、实际应用例子

1、在使用长连接的comet服务端推送技术中,消息推送线程设置为守护线程,服务于ChatServlet的servlet用户线程,servlet的init启动消息线程,servlet一旦初始化后,一直存在于服务器,servlet摧毁后,消息线程自动退出。

2、容器收到一个servlet请求,调度线程从线程池中选出一个工作者线程,然后由该线程来执行servlet的service方法。当这个线程正在执行的时候,容器收到另一个请求,调度线程同样从线程池中选出另一个工作者线程来服务新的请求,容器并不关心这个线程是否访问的是同一个servlet,当容器同时收到同一个servlet的多个请求的时候,name这个servlet的service方法将在多线程中并发执行。

3、servlet容器默认采用单例多线程的方式来处理请求,这样减少产生servlet实例的开销,提升了对请求的响应时间,对tomcat可以在server.xml中通过元素设置线程池中线程的数目。

20130523112456083

四、为什么要用守护线程?

1、我们知道静态变量是classLoader级别的,如果web应用程序停止,这些静态变量也会从JVM中清除。但是线程时JVM级别的,如果在web应用中启动一个线程,这个线程的声明周期并不会和web应用程序保持同步。就是说,即使停止了web应用,这个线程依旧是活跃的。正是因为这个隐晦的问题,所以很多有经验的开发者不赞成在web应用中私自启动线程。

2、如果我们手动使用JDK Timer,在web容器启动时启动Timer,当web容器关闭时,除非手动关闭这个timer,否者timer中的任务还会继续执行。

3、下面通过一个小例子来演示这个“诡异”的现象,我们通过ServletContextListener在Web容器启动时创建一个Timer并周期性地运行一个任务:

  1. //代码清单StartCycleRunTask:容器监听器
  2. package com.baobaotao.web;
  3. import java.util.Date;
  4. import java.util.Timer;
  5. import java.util.TimerTask;
  6. import javax.servlet.ServletContextEvent;
  7. import javax.servlet.ServletContextListener;
  8. public class StartCycleRunTask implements ServletContextListener ...{
  9. private Timer timer;
  10. public void contextDestroyed(ServletContextEvent arg0) ...{
  11. // ②该方法在Web容器关闭时执行
  12. System.out.println("Web应用程序启动关闭...");
  13. }
  14. public void contextInitialized(ServletContextEvent arg0) ...{
  15. //②在Web容器启动时自动执行该方法
  16. System.out.println("Web应用程序启动...");
  17. timer = new Timer();//②-1:创建一个Timer,Timer内部自动创建一个背景线程
  18. TimerTask task = new SimpleTimerTask();
  19. timer.schedule(task, 1000L, 5000L); //②-2:注册一个5秒钟运行一次的任务
  20. }
  21. }
  22. class SimpleTimerTask extends TimerTask ...{//③任务
  23. private int count;
  24. public void run() ...{
  25. System.out.println((++count)+"execute task..."+(new Date()));
  26. }
  27. }

4、在Tomcat中部署这个Web应用并启动后,你将看到任务每隔5秒钟执行一次。
运行一段时间后,登录Tomcat管理后台,将对应的Web应用(chapter13)关闭。

5、转到Tomcat控制台,你将看到虽然Web应用已经关闭,但Timer任务还在我行我素地执行如故——舞台已经拆除,戏子继续表演:

6、我们可以通过改变清单StartCycleRunTask的代码,在contextDestroyed(ServletContextEvent arg0)中添加timer.cancel()代码,在Web容器关闭后手工停止Timer来结束任务。

五、守护线程与spring

spring为jdk timer和Quartz Scheduler提供的timerFactoryBean和ScherdulerFactoryBean能够和spring容器的生命周期关联,在spring容器启动时启动调度器,而在spring容器关闭时,停止调度器。所以在spring中通过这两个factoryBean配置调度器,再从spring IOC中获取调度器引用进行任务调度将不会出现这种web容器关闭而任务依然运行的问题,而如果你在程序中直接使用timer或scheduler,如不进行额外的处理,将会出现类似的问题。

往期精彩内容:

Java知识体系总结(2021版)

Java多线程基础知识总结(绝对经典)

【全栈最全Java框架总结】SSH、SSM、Springboot

超详细的springBoot学习笔记

常见数据结构与算法整理总结

Java设计模式:23种设计模式全面解析(超级详细)

Java面试题总结(附答案)

发表评论

表情:
评论列表 (有 0 条评论,587人围观)

还没有评论,来说两句吧...

相关阅读

    相关 线守护线

    一、守护线程介绍 线程分为用户线程和守护线程,而守护线程就是为用户线程服务的一种线程,一般具有较低的优先级,用户线程停止,守护线程也就停止了。 举个例子:你在看电视,那

    相关 Java线守护线

    守护线程在没有用户线程可服务时自动离开,在Java中比较特殊的线程是被称为守护(Daemon)线程的低级别线程。这个线程具有最低的优先级,用于为系统中的其它对象和线程提供服务。

    相关 线-守护线

    守护线程的概念是: 当一个进程里,所有的线程都是守护线程的时候,结束当前进程。 就好像一个公司有销售部,生产部这些和业务挂钩的部门。 除此之外,还有后勤,行政等这