`
talentluke
  • 浏览: 592184 次
  • 性别: Icon_minigender_1
  • 来自: 深圳
社区版块
存档分类
最新评论

后台线程(守护线程)

阅读更多

所谓的后台线程,是指在程序运行的时候在后台提供一种通用服务的线程,并且这种线程并不属于程序中不可或缺的部分。因此当所有的非后台线程结束时,程序也就终止了,同时会杀死所有后台线程。反过来说,只要有任何非后台线程(用户线程)还在运行,程序就不会终止。后台线程在不执行finally子句的情况下就会终止其run方法后台线程创建的子线程也是后台线程。

下面是一个后台线程的示例:

[java] view plaincopy
 
  1. <span style="font-size:16px;">package demo.thread;  
  2.   
  3. import java.util.concurrent.TimeUnit;  
  4.   
  5. public class DaemonDemo implements Runnable {  
  6.     @Override  
  7.     public void run() {  
  8.         try {  
  9.             while (true) {  
  10.                 Thread.sleep(1000);  
  11.                 System.out.println("#" + Thread.currentThread().getName());  
  12.             }  
  13.         } catch (InterruptedException e) {  
  14.             e.printStackTrace();  
  15.         } finally {// 后台线程不执行finally子句  
  16.             System.out.println("finally ");  
  17.         }  
  18.     }  
  19.   
  20.     public static void main(String[] args) {  
  21.         for (int i = 0; i < 10; i++) {  
  22.             Thread daemon = new Thread(new DaemonDemo());  
  23.             // 必须在start之前设置为后台线程  
  24.             daemon.setDaemon(true);  
  25.             daemon.start();  
  26.         }  
  27.         System.out.println("All daemons started");  
  28.         try {  
  29.             TimeUnit.MILLISECONDS.sleep(1000);  
  30.         } catch (InterruptedException e) {  
  31.             // TODO Auto-generated catch block  
  32.             e.printStackTrace();  
  33.         }  
  34.     }  
  35. }  
  36. </span>  


 

运行结果:

All daemons started
#Thread-2
#Thread-3
#Thread-1
#Thread-0
#Thread-9
#Thread-6
#Thread-8
#Thread-5
#Thread-7
#Thread-4

分析:从结果可以看出,十个子线程并没有无线循环的打印,而是在主线程(main())退出后,JVM强制关闭所有后台线程。而不会有任何希望出现的确认形式,如finally子句不执行。

 

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

用个比较通俗的比如,任何一个守护线程都是整个JVM中所有非守护线程的保姆:

只要当前JVM实例中尚存在任何一个非守护线程没有结束,守护线程就全部工作;只有当最后一个非守护线程结束时,守护线程随着JVM一同结束工作。
Daemon的作用是为其他线程的运行提供便利服务,守护线程最典型的应用就是 GC (垃圾回收器),它就是一个很称职的守护者。

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


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

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


这里有几点需要注意: 

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

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

 

[java] view plaincopy
 
  1. //完成文件输出的守护线程任务  
  2. import java.io.*;     
  3.     
  4. class TestRunnable implements Runnable{     
  5.     public void run(){     
  6.                try{     
  7.                   Thread.sleep(1000);//守护线程阻塞1秒后运行     
  8.                   File f=new File("daemon.txt");     
  9.                   FileOutputStream os=new FileOutputStream(f,true);     
  10.                   os.write("daemon".getBytes());     
  11.            }     
  12.                catch(IOException e1){     
  13.           e1.printStackTrace();     
  14.                }     
  15.                catch(InterruptedException e2){     
  16.                   e2.printStackTrace();     
  17.            }     
  18.     }     
  19. }     
  20. public class TestDemo2{     
  21.     public static void main(String[] args) throws InterruptedException     
  22.     {     
  23.         Runnable tr=new TestRunnable();     
  24.         Thread thread=new Thread(tr);     
  25.                 thread.setDaemon(true); //设置守护线程     
  26.         thread.start(); //开始执行分进程     
  27.     }     
  28. }     
  29. //运行结果:文件daemon.txt中没有"daemon"字符串。  


看到了吧,把输入输出逻辑包装进守护线程多么的可怕,字符串并没有写入指定文件。原因也很简单,直到主线程完成,守护线程仍处于1秒的阻塞状态。这个时候主线程很快就运行完了,虚拟机退出,Daemon停止服务,输出操作自然失败了。

 

 

[java] view plaincopy
 
  1. public class Test {  
  2.   public static void main(String args) {  
  3.   Thread t1 = new MyCommon();  
  4.   Thread t2 = new Thread(new MyDaemon());  
  5.   t2.setDaemon(true); //设置为守护线程  
  6.   t2.start();  
  7.   t1.start();  
  8.   }  
  9.   }  
  10.   class MyCommon extends Thread {  
  11.   public void run() {  
  12.   for (int i = 0; i < 5; i++) {  
  13.   System.out.println("线程1第" + i + "次执行!");  
  14.   try {  
  15.   Thread.sleep(7);  
  16.   } catch (InterruptedException e) {  
  17.   e.printStackTrace();  
  18.   }  
  19.   }  
  20.   }  
  21.   }  

 

[html] view plaincopy
 
  1. class MyDaemon implements Runnable {  
  2.   public void run() {  
  3.   for (long i = 0; i < 9999999L; i++) {  
  4.   System.out.println("后台线程第" + i + "次执行!");  
  5.   try {  
  6.   Thread.sleep(7);  
  7.   } catch (InterruptedException e) {  
  8.   e.printStackTrace();  
  9.   }  
  10.   }  
  11.   }  
  12.   }  


后台线程第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判断程序是否执行结束的标准是所有的前台执线程行完毕了,而不管后台线程的状态,因此,在使用后台县城时候一定要注意这个问题。 

补充说明:
定义:守护线程--也称“服务线程”,在没有用户线程可服务时会自动离开。
优先级:守护线程的优先级比较低,用于为系统中的其它对象和线程提供服务。
设置:通过setDaemon(true)来设置线程为“守护线程”;将一个用户线程设置为
守护线程的方式是在 线程对象创建 之前 用线程对象的setDaemon方法。
example: 垃圾回收线程就是一个经典的守护线程,当我们的程序中不再有任何运行的
Thread,程序就不会再产生垃圾,垃圾回收器也就无事可做,所以当垃圾回收线程是
JVM上仅剩的线程时,垃圾回收线程会自动离开。它始终在低级别的状态中运行,用于
实时监控和管理系统中的可回收资源。
生命周期:守护进程(Daemon)是运行在后台的一种特殊进程。它独立于控制终端并且
周期性地执行某种任务或等待处理某些发生的事件。也就是
说守护线程不依赖于终端,但是依赖于系统,与系统“同生共死”。那Java的守护线程是
什么样子的呢。当JVM中所有的线程都是守护线程的时候,JVM就可以退出了;如果还有一个
或以上的非守护线程则JVM不会退出。


实际应用例子:

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

容器收到一个Servlet请求,调度线程从线程池中选出一个工作者线程,将请求传递给该工作者线程,然后由该线程来执行Servlet的 service方法。当这个线程正在执行的时候,容器收到另外一个请求,调度线程同样从线程池中选出另一个工作者线程来服务新的请求,容器并不关心这个请求是否访问的是同一个Servlet.当容器同时收到对同一个Servlet的多个请求的时候,那么这个Servlet的service()方法将在多线程中并发执行。
Servlet容器默认采用单实例多线程的方式来处理请求,这样减少产生Servlet实例的开销,提升了对请求的响应时间,对于Tomcat可以在server.xml中通过<Connector>元素设置线程池中线程的数目。
如图: 

 



为什么要用守护线程?

 

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

如果我们手工使用JDK Timer(Quartz的Scheduler),在Web容器启动时启动Timer,当Web容器关闭时,除非你手工关闭这个Timer,否则Timer中的任务还会继续运行!

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

[java] view plaincopy
 
  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. }  


在web.xml中声明这个Web容器监听器:<?xml version="1.0" encoding="UTF-8"?>
<web-app> 
… 
<listener> 
<listener-class>com.baobaotao.web.StartCycleRunTask</listener-class> 
</listener> 
</web-app> 

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

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

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

Spring为JDK Timer和Quartz Scheduler所提供的TimerFactoryBean和SchedulerFactoryBean能够和Spring容器的生命周期关联,在 Spring容器启动时启动调度器,而在Spring容器关闭时,停止调度器。所以在Spring中通过这两个FactoryBean配置调度器,再从 Spring IoC中获取调度器引用进行任务调度将不会出现这种Web容器关闭而任务依然运行的问题。而如果你在程序中直接使用Timer或Scheduler,如不 进行额外的处理,将会出现这一问题。 

 

分享到:
评论

相关推荐

    ASPNET中实现在线用户检测(使用后台守护线程)

    ASPNET中实现在线用户检测(使用后台守护线程)ASPNET中实现在线用户检测(使用后台守护线程)ASPNET中实现在线用户检测(使用后台守护线程)ASPNET中实现在线用户检测(使用后台守护线程)

    ASPNET中实现在线用户检测(使用后台守护线程).rar

    ASPNET中实现在线用户检测(使用后台守护线程).rar

    Java后台线程操作示例【守护线程】

    主要介绍了Java后台线程操作,结合实例形式分析了java守护线程相关原理、操作技巧与使用注意事项,需要的朋友可以参考下

    Estom#notes#多线程的其他操作1

    多线程 守护线程守护线程就是后台线程,也是一种依赖线程特点:当前台线程结束后,后台线程会自动终止,作为依赖线程,守护线程,不用强调结束。toString 方法能

    用qt实现的程序守护进程程序

    主要功能: 可设置检测的程序名称。 可设置udp通信端口。 可设置超时次数。 自动记录已重启次数。 自动记录最后一次重启时间。...自动隐藏的托盘运行或者后台运行。 提供界面设置程序名称已经开启和暂停服务。

    java基础知识线程讲解和练习

    Java线程是Java语言中一个非常重要的概念,它允许程序同时执行多个任务。...用户线程是程序显式创建和控制的线程,而守护线程是在后台运行的特殊线程,用于执行一些后台任务,如垃圾回收、资源管理等。

    c# 面试必备线程基础知识点

    线程的知识太多,知识点有深有浅,往深的研究会涉及操作系统、CPU、内存,往浅了说就是一些语法。没有一定的知识积累,很难把线程的知识...根据线程运行模式,可以把线程分为前台线程、后台线程和守护(Daemon)线程:

    Python多线程编程篇教程(实例).PDF

    其实创建线程之后,线程并不是始终保持一个状态的,其状态大概如下: * New 创建 * Runnable 就绪。等待调度 ...* 守护线程(后台线程) * 前台线程 简单了解完这些之后,我们开始看看具体的代码使用了。

    守护进程(互相监听)

    没有采用网上流传的哪几种指标不治本的方法,,而是采用底层ndk,jni思想从本质上解决问题的。希望对大家有所帮助。。

    后台开发核心技术与应用实践

    第四部分(第9~11章)主要是多线程、进程和进程间通信相关的知识,包括多线程的使用、多线程的同步和重入问题,进程方面有父子进程、僵死进程、守护进程和进程间通讯的方式。 第五部分(第12章)主要是HTTP协议的...

    后台开发的一些源代码

    第四部分(第9~11章)主要是多线程、进程和进程间通信相关的知识,包括多线程的使用、多线程的同步和重入问题,进程方面有父子进程、僵死进程、守护进程和进程间通讯的方式。 第五部分(第12章)主要是HTTP协议的...

    Linux网络吞吐量多线程基准测试工具。-C/C++开发

    NTTTCP-for-Linux概述一个多线程...支持在后台运行(守护程序,“-D”)。 默认情况下支持发送方和接收方同步模式。 使用“ -N”(no_sync)禁用同步。 支持使用多个客户端模式进行测试(在Receiver上使用“ -M”,

    后台开发 核心技术与应用实践

    第四部分(第9~11章)主要是多线程、进程和进程间通信相关的知识,包括多线程的使用、多线程的同步和重入问题,进程方面有父子进程、僵死进程、守护进程和进程间通讯的方式。 第五部分(第12章)主要是HTTP协议的...

    徐晓鑫后台开发技术实践——腾讯

    第四部分(第9~11章)主要是多线程、进程和进程间通信相关的知识,包括多线程的使用、多线程的同步和重入问题,进程方面有父子进程、僵死进程、守护进程和进程间通讯的方式; 第五部分(第12章)主要是HTTP协议的...

    linux下的守护进程

    守护进程(Daemon)是运行在后台的一种特殊进程。它独立于控制终端并且周期性地执行某种任务或等待处理某些发生的事件。守护进程是一种很有用的进程。Linux的大多数服务器就是用守护进程实现的。同时,守护进程完成...

    控制台守护进程 Symbian

    实现Server的后台时时运行,实现2个线程的相互咬合

    局域网文件共享软件

    主要工作:写了一个前台和用户交互的程序,写了一个后台和其他电脑通信的守护程序。前台负责发送命令到后台,并接收后台返回的信息。后台用消息队列和前台通信,并向远程客户端提供服务。用到主要编程技术有:多线程...

    linux 管道

    这个是linux 多进程多线程通信的模板,多进程间命名管道实现前台进程控制后台守护进程,线程间使用互斥锁机制

    守护进程、脚本、指定外部配置文件

    ①守护进程(daemon)是一类在后台运行的特殊进程,用于执行特定的系统任务。很多守护进程在系统引导的时候启动,并且一直运行直到系统关闭。另一些只在需要的时候才启动,完成任务后就自动结束。 ②这是我在公司用的...

    浅析PHP7的多进程及实例

    我们都知道PHP是单进程执行的,PHP处理多并发主要是依赖服务器或PHP-FPM的多进程及它们进程的复用,但PHP实现多进程也意义重大,尤其是在后台Cli模式下处理大量数据或运行后台DEMON守护进程时,多进程的优势不用多说...

Global site tag (gtag.js) - Google Analytics