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

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

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

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

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来进行服务,比如读写操作或者计算逻辑。

4、代码实例

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

import java.io.File;

import java.io.FileOutputStream;

import java.io.IOException;

public class TestRunnable implements Runnable {

public void run(){

try {

Thread.sleep(1000);

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 static void main(String[] args) {

Runnable tr = new TestRunnable();

Thread thread = new Thread(tr);

thread.setDaemon(true);

thread.start();

}

}

//运行结果:文件daemon.txt中没有"daemon"字符串。

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

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

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

代码实例:

package reflect;

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();

}

}

package reflect;

public class MyDaemon implements Runnable{

public void run(){

for (int i = 0; i < 1000; i++) {

System.out.println("后台线程第"+i+"次执行!");

try {

Thread.sleep(7);

} catch (InterruptedException e) {

e.printStackTrace();

}

}

}

}

package reflect;

public 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();

}

}

}

}

运行结果:

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

前台线程是保证执行完毕的,后台线程还没有执行完毕就退出了。

实际上: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中通过元素设置线程池中线程的数目。

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

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

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

3、下面通过一个小例子来演示这个“诡异”的现象,我们通过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()));

}

}

4、在Tomcat中部署这个Web应用并启动后,你将看到任务每隔5秒钟执行一次。

运行一段时间后,登录Tomcat管理后台,将对应的Web应用(chapter13)关闭。

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

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

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

java守护线程的使用场景_浅谈Java守护线程相关推荐

  1. java接口与类相同不同_浅谈java的接口和C++虚类的相同和不同之处

    C++虚类相当于java中的抽象类,与接口的不同之处是: 1.一个子类只能继承一个抽象类(虚类),但能实现多个接口 2.一个抽象类可以有构造方法,接口没有构造方法 3.一个抽象类中的方法不一定是抽象方 ...

  2. java中修饰常量的事_浅谈java中的声明常量为什么要用static修饰

    今天定义一个类常量,想着也只有这个类可以用到,就没用static关键字修饰.结果sonar代码检查提示: Rename this field "PERSON_TYPE_USER" ...

  3. java中的标识符和关键字_浅谈java中的标识符、修饰符和关键字

    合法标识符 Java语言中,对于变量,常量,函数,语句块均有名字,我们统统称之为Java标识符.标识符是用来给类.对象.方法.变量.接口和自定义数据类型命名的. 组成:Java标识符由数字,字母和下划 ...

  4. java程序的装载与检查_浅谈Java类型装载、连接与初始化

    类型装载.连接与初始化 Java虚拟机通过装载.连接和初始化一个Java类型,使该类型可以被正在运行的Java程序所使用.其中装载就是把二进制形式的Java class文件读入Java虚拟机中去;连接 ...

  5. java为什么要分代回收_浅谈Java堆内存分代回收

    1.概述 与C++不同的是, 在Java中我们无需关心对象占用空间的释放, 这主要得益于Java中的垃圾处理器(简称GC)帮助我们自动的进行对象占用空间的释放. 下面我们带着几个问题来学习: 堆内存是 ...

  6. java final类为什么不能继承_浅谈Java之终止继承:Final类和Fianl方法

    前言 关键字Final不仅可以用来修饰变量,而且对类及其方法的继承也有很大的影响,本文将从类与方法两个方面介绍final关键字的功能. Final类 当关键字final用来修饰类时,其含义是该类不能再 ...

  7. java合并单元格的快捷键_浅谈java单元格合并的实现(转载)

    摘要:本文主要叙述如何通过建立数据库模型(单元格模型).重载JTable的三个重要方法(getCellRect(),columnAtPoint(),rowAtPoint())和继承Jtable的渲染组 ...

  8. 简述java中流的概念和划分_浅谈Java中流的概念与用途

    Java所有的I/O机制都是基于数据流进行输入输出,这些数据流表示了字符或者字节数据的流动序列.Java的I/O流提供了读写数据的标准方法.任何Java中表示数据源的对象都会提供以数据流的方式读写它的 ...

  9. java bitset用途_浅谈Java BitSet使用场景和代码示例

    搜索热词 @H_502_0@一.什么是BitSet? @H_502_0@ 注:以下内容来自JDK API: @H_502_0@ BitSet类实现了一个按需增长的位向量.位Set的每一个组件都有一个b ...

最新文章

  1. Linux设置默认Python版本
  2. C++字符串详解(三) 字符串的增删改
  3. (04741)计算机网络原理,04741计算机网络原理真题_成都英才教育网
  4. 笔记-高项案例题-2014年上-计算题
  5. 洛谷 1858 多人背包
  6. datetimepicker不可以选择当天之前_专访吴京:网上《战狼3》的消息我都不知道,大家可以选择不信...
  7. 使用windows system backup 裸机恢复域控-windows2008
  8. Oracle11g exp 1445,oracle11g 导出表报EXP-00011:table不存在。
  9. 智能硬件产品开发分享
  10. sql 获取当前日期的季度,年份,月份等日期部分
  11. 会计需要哪方面的计算机知识,学会计需要哪些知识
  12. cat6 万兆_CAT6、CAT6A超六类和CAT7七类网线哪个更适合10GBASE-T万兆网络
  13. android hid 触摸屏驱动,针对USB接口触摸屏HID免驱触摸屏配置方法
  14. 三极管在ad中的原理图库_Altium Designer原理图元器件库详细说明
  15. July大神交大读书会子atoi
  16. 机械硬盘数据迁移到固态硬盘怎么操作?
  17. 计算机打开硬盘响应慢,电脑处理多任务卡顿,开机反应慢,换固态硬盘不能解决问题根本...
  18. 语义通信(Semantic Communication)
  19. 网投简历应该注意些什么
  20. 安卓手机怎么设置蓝牙耳机弹窗动画_链接重推其他团无线蓝牙耳机

热门文章

  1. day6 (常用模块和面向对象基础)
  2. winmgmt服务不存在或已标记为删除·windows management instrumentation这个服务无法启动...
  3. 飞歌车机升级LOGO篇
  4. html5购票代码,HTML5代码大全
  5. Android studio 中R文件出错解决办法
  6. java项目总监_写给我们项目总监的一些话
  7. 零距离泛目录程序源码主动关键词主动收集主动配图 原创
  8. 通过linux课程我学到了什么作文,我学会了什么作文(精选3篇)
  9. Qt程序组播接收不到数据
  10. dropbox连不上_如何在Android和iOS上清除Dropbox缓存