所谓的后台线程,是指在程序运行的时候在后台提供一种通用服务的线程,并且这种线程并不属于程序中不可或缺的部分。因此当所有的非后台线程结束时,程序也就终止了,同时会杀死所有后台线程。反过来说,只要有任何非后台线程(用户线程)还在运行,程序就不会终止。后台线程在不执行finally子句的情况下就会终止其run方法。后台线程创建的子线程也是后台线程。
下面是一个后台线程的示例:
- <span style="font-size:16px;">package demo.thread;
- import java.util.concurrent.TimeUnit;
- public class DaemonDemo implements Runnable {
- @Override
- public void run() {
- try {
- while (true) {
- Thread.sleep(1000);
- System.out.println("#" + Thread.currentThread().getName());
- }
- } catch (InterruptedException e) {
- e.printStackTrace();
- } finally {// 后台线程不执行finally子句
- System.out.println("finally ");
- }
- }
- public static void main(String[] args) {
- for (int i = 0; i < 10; i++) {
- Thread daemon = new Thread(new DaemonDemo());
- // 必须在start之前设置为后台线程
- daemon.setDaemon(true);
- daemon.start();
- }
- System.out.println("All daemons started");
- try {
- TimeUnit.MILLISECONDS.sleep(1000);
- } catch (InterruptedException e) {
- // TODO Auto-generated catch block
- e.printStackTrace();
- }
- }
- }
- </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也就没有工作可做了,也就没有继续运行程序的必要了。
值得一提的是,守护线程并非只有虚拟机内部提供,用户在编写程序时也可以自己设置守护线程。下面的方法就是用来设置守护线程的。
- 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判断程序是否执行结束的标准是所有的前台执线程行完毕了,而不管后台线程的状态,因此,在使用后台县城时候一定要注意这个问题。
补充说明:
定义:守护线程--也称“服务线程”,在没有用户线程可服务时会自动离开。
优先级:守护线程的优先级比较低,用于为系统中的其它对象和线程提供服务。
设置:通过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并周期性地运行一个任务:
- //代码清单StartCycleRunTask:容器监听器
- package com.baobaotao.web;
- import java.util.Date;
- import java.util.Timer;
- import java.util.TimerTask;
- import javax.servlet.ServletContextEvent;
- import javax.servlet.ServletContextListener;
- public class StartCycleRunTask implements ServletContextListener ...{
- private Timer timer;
- public void contextDestroyed(ServletContextEvent arg0) ...{
- // ②该方法在Web容器关闭时执行
- System.out.println("Web应用程序启动关闭...");
- }
- public void contextInitialized(ServletContextEvent arg0) ...{
- //②在Web容器启动时自动执行该方法
- System.out.println("Web应用程序启动...");
- timer = new Timer();//②-1:创建一个Timer,Timer内部自动创建一个背景线程
- TimerTask task = new SimpleTimerTask();
- timer.schedule(task, 1000L, 5000L); //②-2:注册一个5秒钟运行一次的任务
- }
- }
- class SimpleTimerTask extends TimerTask ...{//③任务
- private int count;
- public void run() ...{
- System.out.println((++count)+"execute task..."+(new Date()));
- }
- }
在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,如不 进行额外的处理,将会出现这一问题。
相关推荐
根据提供的信息,我们可以详细探讨如何在ASP.NET中实现在线用户检测功能,并且通过使用后台守护线程来确保这一过程能够持续有效地运行。 ### ASP.NET中实现在线用户检测 #### 背景介绍 在线用户检测是一项重要的...
创建守护线程的常见应用场景包括后台服务、资源清理、日志记录等。这些任务不直接影响程序的主要业务逻辑,但对程序的健康运行至关重要。例如,一个守护线程可能负责定期检查系统的内存使用情况,并在内存不足时释放...
Java后台线程操作示例【守护线程】 Java后台线程操作示例【守护线程】是Java多线程编程中的一种重要概念,主要介绍了Java后台线程操作的原理、操作技巧与使用注意事项。下面我们将详细介绍Java后台线程操作示例...
本教程将详细讲解如何利用后台守护线程来完成这一任务。守护线程是一种特殊的线程,它在应用程序启动时创建,并在所有其他非守护线程结束后继续运行,直到应用程序关闭。在ASP.NET环境中,我们可以巧妙地运用守护...
守护线程是一种特殊类型的线程,当所有非守护线程结束时,即使还有守护线程在运行,程序也会终止。 总结来说,进程与线程是操作系统并发执行任务的基础。理解它们的工作原理,掌握线程同步、前后台线程的概念,以及...
System.out.println("后台线程第" + i + "次执行!"); try { Thread.sleep(7); } catch (InterruptedException e) { e.printStackTrace(); } } } } ``` 在这个例子中,`MyCommon`线程是用户线程,而`...
守护线程是一种长期运行的后台进程,它不依赖于终端会话,即使用户注销或者关闭了终端,守护线程依然可以继续工作。 在本场景中,我们要实现的功能是通过守护线程来检查一个程序(假设为`target_program`)是否正在...
1. 守护线程是运行在程序后台的线程。 2. 守护线程创建的进程依然是守护进程。 3. 守护线程不会影响JVM的退出,当JVM只剩余守护进程时,JVM进行退出。 4. 守护线程在JVM退出时,自动销毁。 在开发Java守护线程时,...
用户线程即运行在前台的线程,而守护线程是运行在后台的线程。守护线程的作用是为其他前台线程的运行提供便利服务,而且仅在普通、非守护线程仍然运行时才需要。 守护线程的特点是,当 JVM 检测仅剩一个守护线程,...
通常,后台服务或者定时任务等不直接影响程序主要功能的线程会被设置为守护线程。 在Java中,我们可以通过`Thread.setDaemon(true)`方法将一个线程设置为守护线程,该方法必须在线程启动前调用。例如: ```java ...
在Java编程语言中,守护线程(Daemon Thread)被称为“变形金刚中的守护神”,这是因为它们在后台默默地工作,为应用程序提供必要的服务,而不会阻止程序的退出。理解并正确使用守护线程是Java多线程编程中的一个...
在上面的代码中,我们创建了一个名为 TestThread 的线程类,并将其设置为后台线程(守护线程)。然后,我们启动了该线程。在 main 方法中,我们不再有任何输出语句,因为后台线程自动退出。 第三个例子:线程 join...
守护线程的实际应用场景很多,例如在web服务器中的Servlet,容器启动时后台初始化一个服务线程。守护线程可以提供一些基础服务,例如GC线程、定时器线程等。守护线程是一种特殊的线程机制,用于提供一些基础服务,而...
Java中的后台线程,也称为守护线程(daemon),在程序运行中扮演着重要的角色,尤其是在需要持续提供服务而不需要直接影响程序生命周期的情况下。本文将深入探讨后台线程的概念、其特性以及如何在Java中创建和管理...
守护线程是用于支持应用程序运行的后台线程,比如垃圾回收器就是守护线程。当所有非守护线程结束时,即使还有守护线程在运行,JVM也会退出。我们可以使用`setDaemon(true)`方法将线程设置为守护线程。 6. 线程...
守护线程,也被称为后台线程,是Java编程语言中的一种特殊线程类型。它们的主要特点是在所有非守护线程(用户线程)结束后,守护线程才会被系统强制结束。在Windows环境下,通常我们使用服务来实现长期运行的任务,...
守护进程和多线程是操作系统中的重要概念,它们在后台执行任务,支持系统服务和应用程序的高效运行。守护进程(daemon)通常不与任何终端关联,独立于控制终端并作为后台进程运行。在Linux系统中,以字母'd'结尾的...
在Java编程中,守护线程(Daemon Thread)是一种特殊的线程类型,它的主要作用是为应用程序提供后台服务,而不会阻止程序的退出。当所有的非守护线程(用户线程)结束时,JVM会自动关闭守护线程并退出。在本项目...
守护线程主要用作程序中后台调度以及支持性工作。当一个 Java 虚拟机中不 存 在 非 Daemon 线 程 的 时 候,Java 虚拟机将退 出。通过 setDaemon(true) 方法可以设置守护线程,但是要在线程启动之前设置。在 Java ...
Java中的守护线程(Daemon Thread)是Java虚拟机(JVM)的一种特殊线程类型,它们在后台执行任务,主要用于为应用程序的其他非守护线程提供服务。守护线程的一个典型例子是垃圾回收器(Garbage Collector),它默默...