作者 | 王磊

来源 | Java中文社群(ID:javacn666)

转载请联系授权(微信ID:GG_Stone)

在 Java 语言中线程分为两类:用户线程和守护线程,而二者之间的区别却鲜有人知,所以本文磊哥带你来看二者之间的区别,以及守护线程需要注意的一些事项。

1.默认用户线程

Java 语言中无论是线程还是线程池,默认都是用户线程,因此用户线程也被称为普通线程。

以线程为例,想要查看线程是否为守护线程只需通过调用 isDaemon() 方法查询即可,如果查询的值为 false 则表示不为守护线程,自然也就属于用户线程了,如下代码所示:

public static void main(String[] args) throws InterruptedException {Thread thread = new Thread(new Runnable() {@Overridepublic void run() {System.out.println("我是子线程");}});System.out.println("子线程==守护线程:" + thread.isDaemon());System.out.println("主线程==守护线程:" + Thread.currentThread().isDaemon());
}

以上程序的执行结果为:

从上述结果可以看出,默认情况下主线程和创建的新线程都为用户线程

PS:Thread.currentThread() 的意思是获取执行当前代码的线程实例。

2.主动修改为守护线程

守护线程(Daemon Thread)也被称之为后台线程或服务线程,守护线程是为用户线程服务的,当程序中的用户线程全部执行结束之后,守护线程也会跟随结束。

守护线程的角色就像“服务员”,而用户线程的角色就像“顾客”,当“顾客”全部走了之后(全部执行结束),那“服务员”(守护线程)也就没有了存在的意义,所以当一个程序中的全部用户线程都结束执行之后,那么无论守护线程是否还在工作都会随着用户线程一块结束,整个程序也会随之结束运行。

那如何将默认的用户线程修改为守护线程呢?

这个问题要分为两种情况来回答,首先如果是线程,则可以通过设置 setDaemon(true) 方法将用户线程直接修改为守护线程,而如果是线程池则需要通过 ThreadFactory 将线程池中的每个线程都为守护线程才行,接下来我们分别来实现一下。

2.1 设置线程为守护线程

如果使用的是线程,可以通过 setDaemon(true) 方法将线程类型更改为守护线程,如下代码所示:

 public static void main(String[] args) throws InterruptedException {Thread thread = new Thread(new Runnable() {@Overridepublic void run() {System.out.println("我是子线程");}});// 设置子线程为守护线程thread.setDaemon(true);System.out.println("子线程==守护线程:" + thread.isDaemon());System.out.println("主线程==守护线程:" + Thread.currentThread().isDaemon());}

以上程序的执行结果为:

2.2 设置线程池为守护线程

要把线程池设置为守护线程相对来说麻烦一些,需要将线程池中的所有线程都设置成守护线程,这个时候就需要使用 ThreadFactory 来定义线程池中每个线程的线程类型了,具体实现代码如下:

// 创建固定个数的线程池
ExecutorService threadPool = Executors.newFixedThreadPool(10, new ThreadFactory() {@Overridepublic Thread newThread(Runnable r) {Thread t = new Thread(r);// 设置线程为守护线程t.setDaemon(false);return t;}
});

如下图所示:

如上图所示,可以看出,整个程序中有 10 个守护线程都是我创建的。其他几种创建线程池的设置方式类似,都是通过 ThreadFactory 统一设置的,这里就不一一列举了。

3.守护线程 VS 用户线程

通过前面的学习我们可以创建两种不同的线程类型了,那二者有什么差异呢?接下来我们使用一个小示例来看一下。

下面我们创建一个线程,分别将这个线程设置为用户线程和守护线程,在每个线程中执行一个 for 循环,总共执行 10 次信息打印,每次打印之后休眠 100 毫秒,来观察程序的运行结果。

3.1 用户线程

新建的线程默认就是用户线程,因此我们无需对线程进行任何特殊的处理,执行 for 循环即可(总共执行 10 次信息打印,每次打印之后休眠 100 毫秒),实现代码如下:

/*** Author:Java中文社群*/
public class DaemonExample {public static void main(String[] args) throws InterruptedException {Thread thread = new Thread(new Runnable() {@Overridepublic void run() {for (int i = 1; i <= 10; i++) {// 打印 i 信息System.out.println("i:" + i);try {// 休眠 100 毫秒Thread.sleep(100);} catch (InterruptedException e) {e.printStackTrace();}}}});// 启动线程thread.start();}
}

以上程序执行结果如下:

从上述结果可以看出,当程序执行完 10 次打印之后才会正常结束进程。

3.2 守护线程

/*** Author:Java中文社群*/
public class DaemonExample {public static void main(String[] args) throws InterruptedException {Thread thread = new Thread(new Runnable() {@Overridepublic void run() {for (int i = 1; i <= 10; i++) {// 打印 i 信息System.out.println("i:" + i);try {// 休眠 100 毫秒Thread.sleep(100);} catch (InterruptedException e) {e.printStackTrace();}}}});// 设置为守护线程thread.setDaemon(true);// 启动线程thread.start();}
}

以上程序执行结果如下:

从上述结果可以看出,当线程设置为守护线程之后,整个程序不会等守护线程 for 循环 10 次之后再进行关闭,而是当主线程结束之后,守护线程只执行了一次循环就结束运行了,由此可以看出守护线程和用户线程的不同。

3.3 小结

守护线程是为用户线程服务的,当一个程序中的所有用户线程都执行完成之后程序就会结束运行,程序结束运行时不会管守护线程是否正在运行,由此我们可以看出守护线程在 Java 体系中权重是比较低的。

4.守护线程注意事项

守护线程的使用需要注意以下三个问题:

  1. 守护线程的设置 setDaemon(true) 必须要放在线程的 start() 之前,否则程序会报错。

  2. 在守护线程中创建的所有子线程都是守护线程。

  3. 使用 jojn() 方法会等待一个线程执行完,无论此线程是用户线程还是守护线程。

接下来我们分别演示一下,以上的注意事项。

4.1 setDaemon 执行顺序

当我们将 setDaemon(true) 设置在 start() 之后,如下代码所示:

public static void main(String[] args) throws InterruptedException {Thread thread = new Thread(new Runnable() {@Overridepublic void run() {for (int i = 1; i <= 10; i++) {// 打印 i 信息System.out.println("i:" + i + ",isDaemon:" +Thread.currentThread().isDaemon());try {// 休眠 100 毫秒Thread.sleep(100);} catch (InterruptedException e) {e.printStackTrace();}}}});// 启动线程thread.start();// 设置为守护线程thread.setDaemon(true);
}

以上程序执行结果如下:

从上述结果可以看出,当我们将 setDaemon(true) 设置在 start() 之后,不但程序的执行会报错,而且设置的守护线程也不会生效。

4.2 守护线程的子线程

public static void main(String[] args) throws InterruptedException {Thread thread = new Thread(new Runnable() {@Overridepublic void run() {Thread thread2 = new Thread(new Runnable() {@Overridepublic void run() {}});System.out.println("守护线程的子线程 thread2 isDaemon:" +thread2.isDaemon());}});// 设置为守护线程thread.setDaemon(true);// 启动线程thread.start();Thread.sleep(1000);
}

以上程序执行结果如下:

从上述结果可以看出,守护线程中创建的子线程,默认情况下也属于守护线程

4.3 join 与守护线程

通过 3.2 部分的内容我们可以看出,默认情况下程序结束并不会等待守护线程执行完,而当我们调用线程的等待方法 join() 时,执行的结果就会和 3.2 的结果有所不同,下面我们一起来看吧,示例代码如下:

public static void main(String[] args) throws InterruptedException {Thread thread = new Thread(new Runnable() {@Overridepublic void run() {for (int i = 1; i <= 10; i++) {// 打印 i 信息System.out.println("i:" + i);try {// 休眠 100 毫秒Thread.sleep(100);} catch (InterruptedException e) {e.printStackTrace();}}}});// 设置为守护线程thread.setDaemon(true);// 启动线程thread.start();// 等待线程执行完thread.join();System.out.println("子线程==守护线程:" + thread.isDaemon());System.out.println("主线程==守护线程:" + Thread.currentThread().isDaemon());
}

以上程序执行结果如下:

通过上述结果我们可以看出,即使是守护线程,当程序中调用 join() 方法时,程序依然会等待守护线程执行完成之后再结束进程。

5.守护线程应用场景

守护线程的典型应用场景就是垃圾回收线程,当然还有一些场景也非常适合使用守护线程,比如服务器端的健康检测功能,对于一个服务器来说健康检测功能属于非核心非主流的服务业务,像这种为了主要业务服务的业务功能就非常合适使用守护线程,当程序中的主要业务都执行完成之后,服务业务也会跟随者一起销毁。

6.守护线程的执行优先级

首先来说,线程的类型(用户线程或守护线程)并不影响线程执行的优先级,如下代码所示,定义一个用户线程和守护线程,分别执行 10 万次循环,通过观察最后的打印结果来确认线程类型对程序执行优先级的影响。

public class DaemonExample {private static final int count = 100000;public static void main(String[] args) throws InterruptedException {// 定义任务Runnable runnable = new Runnable() {@Overridepublic void run() {for (int i = 0; i < count; i++) {System.out.println("执行线程:" + Thread.currentThread().getName());}}};// 创建守护线程 t1Thread t1 = new Thread(runnable, "t1");// 设置为守护线程t1.setDaemon(true);// 启动线程t1.start();// 创建用户线程 t2Thread t2 = new Thread(runnable, "t2");// 启动线程t2.start();}
}

以上程序执行结果如下:

通过上述结果可以看出,线程的类型不管是守护线程还是用户线程对程序执行的优先级是没有任何影响的,而当我们将 t2 的优先级调整为最大时,整个程序的运行结果就完全不同了,如下代码所示:

public class DaemonExample {private static final int count = 100000;public static void main(String[] args) throws InterruptedException {// 定义任务Runnable runnable = new Runnable() {@Overridepublic void run() {for (int i = 0; i < count; i++) {System.out.println("执行线程:" + Thread.currentThread().getName());}}};// 创建守护线程 t1Thread t1 = new Thread(runnable, "t1");// 设置为守护线程t1.setDaemon(true);// 启动线程t1.start();// 创建用户线程 t2Thread t2 = new Thread(runnable, "t2");// 设置 t2 的优先级为最高t2.setPriority(Thread.MAX_PRIORITY);// 启动线程t2.start();}
}

以上程序执行结果如下:

通过上述的结果可以看出,程序的类型和程序执行的优先级是没有任何关系,当新创建的线程默认的优先级都是 5 时,无论是守护线程还是用户线程,它们执行的优先级都是相同的,当将二者的优先级设置不同时,执行的结果也会随之改变(优先级设置的越高,最早被执行的概率也越大)。

7.总结

在 Java 语言中线程分为用户线程和守护线程,守护线程是用来为用户线程服务的,当一个程序中的所有用户线程都结束之后,无论守护线程是否在工作都会跟随用户线程一起结束。守护线程从业务逻辑层面来看权重比较低,但对于线程调度器来说无论是守护线程还是用户线程,在优先级相同的情况下被执行的概率都是相同的。守护线程的经典使用场景是垃圾回收线程,守护线程中创建的线程默认情况下也都是守护线程。

关注公号「Java中文社群」查看更多有意思、涨知识的并发编程文章。

原创不易,点个赞再走呗~

额!Java中用户线程和守护线程区别这么大?相关推荐

  1. java中的后台线程、前台线程、守护线程区别

    java中的后台线程.前台线程.守护线程区别 区别和联系 区别 联系 区别和联系 区别 后台线程和守护线程是一样的. 后台线程不会阻止进程的终止,而前台线程会, 可以在任何时候将前台线程修改为后台线程 ...

  2. java并发:初探用户线程和守护线程

    用户线程和守护线程 用户线程 用户线程执行完,jvm退出.守护线程还是可以跑的 /*** A <i>thread</i> is a thread of execution in ...

  3. JAVA笔记13__创建线程/线程休眠/等待线程终止/线程中断/守护线程

    /*** 线程:是进程的一个执行路径,共享一个内存空间,线程之间可以自由切换,并发执行,一个进程最少有一个进程(单线程程序)* 多线程两种实现方法:1.继承Thread类 2.实现Runnable接口 ...

  4. Java线程之守护线程(Daemon)

    http://blog.csdn.net/mq612/article/details/1520571 守护线程(Daemon) Java有两种Thread:"守护线程Daemon" ...

  5. Java线程之守护线程(Daemon) .

    http://blog.csdn.net/mq612/article/details/1520571 守护线程(Daemon) Java有两种Thread:"守护线程Daemon" ...

  6. JAVA守护线程 非守护线程

    笔记: 第一篇转载写的比较好,将守护线程同linux的守护进程概念进行了对比. 当非守护线程执行完jvm就退出,不管是否还有守护线程在执行.所以守护线程尽量不要执行逻辑代码,顶多执行一些可有可无的辅助 ...

  7. 彻底读懂用户线程和守护线程(代码详解)

    [辰兮要努力]:hello你好我是辰兮,很高兴你能来阅读,昵称是希望自己能不断精进,向着优秀程序员前行! 博客来源于项目以及编程中遇到的问题总结,偶尔会有读书分享,我会陆续更新Java前端.后台.数据 ...

  8. linux非守护线程一直不释放,Linux pthread 和 java thread 的是 / 非守护线程的行为

    Linux pthread 和 java thread 的是 / 非守护线程的行为 pthread_xxx 的函数并没有直接提供设置一个 pthread 为守护线程的 API 而 pthread_at ...

  9. JAVA中的多线程(八):线程的优先级和yield方法

    JAVA中的多线程(八):线程的优先级和yield方法 优先级代表着抢资源的频率 所有线程默认优先级是5 yield()临时释放线程的执行权 1 class Demo implements Runna ...

最新文章

  1. 软件测试2019:第七次作业—— 用户体验测试
  2. JavaEE Tutorials (24) - 资源适配器示例
  3. 【学习笔记】关于正整数除法下取整和上取整的一些基本运算公式
  4. echarts - 使用echarts过程中遇到的问题(pending...)
  5. 宠了4年的老婆,说走就走,没有一点情份,你会怎么做
  6. SharpDeveloeper开发ASP.NET MVC汗流浃背
  7. 【C++学习笔记五】模板
  8. 下面这些是什么意思:@classmethod, @staticmethod, @property?
  9. android 手机无线投屏,安卓手机无线投屏问与答
  10. 无需重装系统,Windows Server 2019系统硬盘无损从MBR转换为GPT格式
  11. 2010 我的求职经历(2)
  12. 【愿头发与你我同在队】团队项目第一次作业-组队与选题
  13. NUC140之I2C和AT24C32
  14. 12.12 生日快乐
  15. 证券运维外包第3个月工作总结
  16. 一个在图片上写字的方法
  17. function函数的基本方法:
  18. 复旦大学计算机科学技术学院院长姜育刚:人工智能算法治理|达观WAIC回顾
  19. c语言符号运算优先级6,c语言运算符号的优先级
  20. access无法 dolby_Windows10专业版无法安装dolby如何解决

热门文章

  1. 用python程序编写二元多项式_Python多项式回归的实现方法
  2. Spark-大规模数据处理计算引擎
  3. 第一章、第一节 Angular基础
  4. 微信小程序开发 | 官方问答精选
  5. drbd(三):drbd的状态说明
  6. 6、EIGRP配置实验之负载均衡
  7. BlockingQueue详解
  8. Mac OSX 安装nvm(node.js版本管理器)
  9. JAVA学习--集合的遍历
  10. 同一个闭区间上有界变差函数的和与积都是有界变差函数