1.如下代码:

package com.bawei.multithread;

public class Recursive {

	private static int counter = 0;

	public static void main(String[] args) {
Thread t = new Thread(){
@Override
public void run() {
try {
//Thread-0线程
System.out.println(Thread.currentThread().getName()+" running");
Thread.sleep(100000);
System.out.println(Thread.currentThread().getName()+" done");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
};
t.start();
try {
Thread.sleep(50000);
//mian线程
System.out.println(Thread.currentThread().getName());
} catch (InterruptedException e) {
e.printStackTrace();
} }
}

当大家运行这段代码的时候,可以通过JDK安装的bin目录下的jconsole工具来查看,当前运行的线程,如下所示:

从这里大家可以看到程序当前运行的线程是main线程和Thread-0线程!

这里可以看到当我们的main线程执行完毕之后,我们的应用程序并没有推出,这是为什么呢?这是因为此时我们的应用程序还有活跃线程Thread-0!

如果我们对程序做一点改动,给线程Thread-0对象设置一个参数即:t.setDaemon(true);此时如果我的main线程结束之后,那么创建的main线程创建的守护线程的生命周期也就跟着结束了!代码如下所示:

package com.bawei.multithread;

public class Recursive {

	private static int counter = 0;

	public static void main(String[] args) {
Thread t = new Thread(){
@Override
public void run() {
try {
//Thread-0线程
System.out.println(Thread.currentThread().getName()+" running");
Thread.sleep(100000);
System.out.println(Thread.currentThread().getName()+" done");
} catch (InterruptedException e) {
e.printStackTrace();
}
}
};
t.setDaemon(true);
t.start();
try {
Thread.sleep(50000);
//mian线程
System.out.println(Thread.currentThread().getName());
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}

而且这里有一点需要特别注意:t.setDaemon(true); 一定要设置在t.start()之前,否则也会报错!

那这个Daemon线程有什么用处呢?

比如在我们建立网络连接的时候,例如从A到B我们建立了一个长连接,大家都知道长连接是要发心跳的,无论是每隔3秒,30秒,1分钟,这个时间是由你自己设置的,然后每隔这段时间客户端就往服务器端发一个心跳包,这样就可以知道对方的存在,就知道对方是可用的,如果不可用可能就需要重新发送连接请求!如果当前线程创建好了一个连接之后,除了利用这个连接通信之外,我还要维护一个心跳,这个心跳和业务逻辑,和通信是没关系的,这个心跳只不过是做维护的一个东西,那这个时候我把创建连接之后的这个线程开辟一个daemon线程,我创建这个daemon线程的目的就是做心跳检查(healthcheck),那这个有什么好处呢?如果我们不这么做,可能建立连接的线程已经死了,但是你的这个维护线程并不会死,必须你去通知它,或者手动关掉该维护线程,但是用stop方法关闭的时候,我们可能也是关闭不了的,因为这个方法已经被弃用了,而且此时如果你的维护线程不关闭,那么你整个应用程序(application)的进程就无法关闭,因为有活活动的线程,如果我们这里将这个维护的线程设置为通信线程的守护线程,那就很好解决了,此时一旦通信线程死掉,那维护线程自然就该挂了,所以守护线程可以帮我们做一些辅助性的东西!

setDaemon()方法默认是设置为false的!

我这里有两个问题,就是这里的Thread-0线程是一个守护线程,如果我在Thread-0这个线程中再开辟一个线程,然后将新开辟的线程也设置为守护线程,

第一:当Thread-0线程关闭的时候,这个新开辟的线程会关掉么?

1.当Thread-0不是守护线程的时候,如果Thread-0执行完毕之后,Thread-0创建的新守护线程会退出么,也就是整个应用程序会结束么?答案是会的,如下代码:

package com.bawei.multithread;

public class Recursive {

	private static int counter = 0;

	public static void main(String[] args) {
//新线程对象
Thread t = new Thread(){
@Override
public void run() {
//在新线程里面又创建了一个新的线程对象,
Thread innerThread = new Thread(){
@Override
public void run() {
try {
Thread.sleep(100*1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}; try {
innerThread.setDaemon(true);
innerThread.start();
Thread.sleep(10*1000);
System.out.println("外层t线程休息好了");
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
};//thread-0 end!
t.start(); }
}

从上面的运行代码可以看出,如果t线程结束之后,t线程创建的守护线程也会结束!

如果我们这里进一步将t线程也设置为守护线程,当main线程结束的时候,t线程毫无疑问会退出,那么t线程创建的新守护线程会退出么?也就是整个应用程序会随着main线程的结束而结束么?答案也是会的。看如下代码演示:

package com.bawei.multithread;

public class Recursive {

	private static int counter = 0;

	public static void main(String[] args) {
//新线程对象
Thread t = new Thread(){
@Override
public void run() {
//在新线程里面又创建了一个新的线程对象,
Thread innerThread = new Thread(){
@Override
public void run() {
try {
Thread.sleep(100*1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
}; try {
innerThread.setDaemon(true);
innerThread.start();
Thread.sleep(10*1000);
System.out.println("外层t线程休息好了");
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
};//thread-0 end!
t.setDaemon(true);
t.start();
System.out.println("main线程结束了");
}
}

所以综上所述:只要是父线程死掉之后,守护线程都会立即死掉!整个应用程序也就结束了!

如果我们这里将t.setDaemon设置为false,也就是默认情况,那么当main线程死掉的时候,实际上子线程就不会死掉了,因为此时的子线程不是守护线程!

package com.bawei.multithread;

public class Recursive {

	private static int counter = 0;

	public static void main(String[] args) {
//新线程对象
Thread t = new Thread(){
@Override
public void run() {
//在新线程里面又创建了一个新的线程对象,
Thread innerThread = new Thread(){
@Override
public void run() {
try {
Thread.sleep(100*1000);
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
} }; try {
innerThread.setDaemon(true);
innerThread.start();
Thread.sleep(10*1000);
System.out.println("外层t线程休息好了");
} catch (InterruptedException e) {
// TODO Auto-generated catch block
e.printStackTrace();
}
}
};//thread-0 end!
//t.setDaemon(true);
t.start();
System.out.println("main线程结束了");
}
}

 运行一下可以看到,如果main线程死掉,此时的t线程也不会挂掉!

第二个问题是如果Thread-0线程执行完毕、main线程也执行完毕,那么整个应用程序(application)会结束么?

我们这里验证一下:

6.Daemon线程的更多相关文章

  1. 【转】关于Java的Daemon线程的理解

    原文地址:http://www.cnblogs.com/ChrisWang/archive/2009/11/28/1612815.html 关于Java的Daemon线程的理解 网上对Java的Dae ...

  2. java concurrency: daemon线程

    daemon线程的概念 在学习操作系统概念的时候,我们就曾听说过daemon的概念.daemon本身指的是在后台运行的进程或者线程,一般用来提供某些不需要与用户直接交互的服务,有点像我们见到的一些系统 ...

  3. [Python]Threading.Thread之Daemon线程

    之前对Daemon线程理解有偏差,特记录说明: 一.什么是Daemon A thread can be flagged as a "daemon thread". The sign ...

  4. python之daemon线程

    [python之daemon线程] A thread can be flagged as a “daemon thread”. The significance of this flag is tha ...

  5. java线程基础知识----java daemon线程

    java线程是一个运用很广泛的重点知识,我们很有必要了解java的daemon线程. 1.首先我们必须清楚的认识到java的线程分为两类: 用户线程和daemon线程 A. 用户线程: 用户线程可以简 ...

  6. Java 学习笔记之 Daemon线程

    Daemon线程: 线程: 用户线程 守护线程 守护线程是一种特殊的线程,在进程中不存在非守护线程了,则守护线程自动销毁. public class DaemonThread extends Thre ...

  7. java线程基础巩固---Daemon线程的创建以及使用场景分析

    daemon线程既守护线程,而在jdk中对于Thread中针对守护线程有专门的API,如下: 而之前在公司项目中就看到过有人使用过Thread中的这个API,但是对于它的使用场景完全不知,所以这次好好 ...

  8. 什么是 Daemon 线程?它有什么意义?

    所谓后台(daemon)线程,是指在程序运行的时候在后台提供一种通用服务的线 程,并且这个线程并不属于程序中不可或缺的部分.因此,当所有的非后台线程 结束时,程序也就终止了,同时会杀死进程中的所有后台 ...

  9. JAVA DAEMON线程的理解

    java线程分两种:用户线程和daemon线程.daemon线程或进程就是守护线程或者进程,但是java中所说的daemon线程和linux中的daemon是有一点区别的. linux中的daemon ...

随机推荐

  1. iOS开发使用pdf切图

    把pdf资源拖到Assets.xcassets里面, 打开最右边的按钮, scales选择single scale就可以像以前一样使用了:  [UIImage imageName:@"xxx ...

  2. java之String类在堆栈存储机制

    String类是一个比较特殊的类,最主要的体现是它有多种创建形式,例如,String a ="abc";Sting a=new("abc");表面上看得到的结果 ...

  3. 【python基础】迭代器和生成器函数

    1.迭代器协议: 1.迭代器协议是指:对象必须提供一个 __next__() 方法,执行该方法要么返回迭代中的下一项,要么就引起一个StopIteration异常,以终止迭代(只能往后走不能往前退) ...

  4. IAM:亚马逊访问权限控制

    IAM的策略.用户->服务器(仓库.业务体) IAM:亚马逊访问权限控制(AWS Identity and Access Management )IAM使您能够安全地控制用户对 AWS 服务和资 ...

  5. (4.4)mysql备份还原——备份存储容灾基础知识

    存储知识 1.为什么需要存储,存储一般解决哪些问题? 容量.速度.易于管理.安全(容灾与备份).可扩展性 2.存储发展历史 [2.1]大型机 [2.2]c/s结构(客户端->服务器) [2.3] ...

  6. (1.5)MySQL表的5种分区类型

    (1.5)MySQL表的5种分区类型 关键词:mysql表分区    小细节: (1)Null 将会放入最小范围区间 (2)无论哪种类型的分区,要么没有主键.唯一键,要么分区表的主键/唯一键都必须包含 ...

  7. ETL : kettle Spoon 转换 + 作业

    Kettle能做什么? 前言 : 需将db2中数据导入到mysql中,利用etl工具进行多表转换.以此为切入点,系统整理.学习kettle工具. 提醒: kettle是纯java编写,机器需要有jre ...

  8. Wordpress安装Redis为网站加速

    前面我们讲了宝塔Linux面板安装Redis,现在我们来举一些例子来看看redis的实际运用,比如Wordpress安装Redis为网站加速,下面就跟着ytkah一起来操作一下. 第一,下载predi ...

  9. 启动Jmeter4.0 后弹出命令窗口提示信息是什么意思?

    启动Jmeter4.0 后弹出命令窗口提示信息: =========================================================================== ...

  10. RN-系列

    8081端口被占用,McAfee Agent关不掉 sudo lsof -n -i4TCP:8081 sudo launchctl list | grep 5693 sudo launchctl re ...