高并发编程-Daemon Thread的创建以及使用场景分析

简介: 高并发编程-Daemon Thread的创建以及使用场景分析

20191031000606569.png

20190918214151115.jpg


官方文档

我们以JAVA8的doc为例 戳这里



20190916233054184.png


Daemon Thread VS User Thread


Java提供两种类型的线程:用户线程和守护程序线程。

  • 用户线程是高优先级线程。 JVM将在终止任务之前等待任何用户线程完成其任务。
  • 守护程序线程是低优先级线程, 其唯一作用是为用户线程提供服务


由于守护程序线程旨在为用户线程提供服务,并且仅在用户线程运行时才需要,因此一旦所有用户线程完成执行,它们都不会阻止JVM退出。


这也就是为什么通常存在于守护程序线程中的无限循环不会导致问题,因为任何代码(包括finally块)都不会在所有用户线程完成执行后执行。因此,不建议将守护程序线程用于I / O任务。


但是,这条规则有例外。守护程序线程中设计糟糕的代码可能会阻止JVM退出。例如,在正在运行的守护程序线程上调用Thread.join()可以阻止应用程序的关闭。


Daemon thread的特点


  • 当所有用户线程完成执行时,它们无法阻止JVM退出。
  • 当所有用户线程完成执行时,JVM会自行终止
  • 如果JVM发现正在运行的守护程序线程,它将终止该线程并在该关闭后自行终。 JVM不关心守护程序线程是否正在运行。
  • 这是一个极低优先级的线程。


方法

void setDaemon(boolean status)

public final void setDaemon(boolean on)
parameters:
on : if true, marks this thread as a daemon thread.
exceptions:
IllegalThreadStateException: if only this thread is active.
SecurityException: if the current thread cannot modify this thread.



此方法用于将当前线程标记为守护程序线程或用户线程。

举个例子:

如果有一个用户线程tU,那么tU.setDaemon(true)会使它成为守护程序线程

如果有一个守护程序线程tD,那么通过调用tD.setDaemon(false)会使它成为用户线程。


boolean isDaemon()


public final boolean isDaemon()
returns: 
This method returns true if this thread is a daemon thread;
false otherwise


此方法用于检查当前是守护进程。 如果线程是守护进程,则返回true,否则返回false。


Exceptions in Daemon thread

如果在启动线程后调用setDaemon()方法,则会抛出IllegalThreadStateException。

package com.artisan.test;
public class DaemonThread extends Thread {
    public void run()
    {
        System.out.println("Thread name: " + Thread.currentThread().getName());
        System.out.println("Check if its DaemonThread: "
                + Thread.currentThread().isDaemon());
    }
    public static void main(String[] args)
    {
        DaemonThread t1 = new DaemonThread();
        DaemonThread t2 = new DaemonThread();
        t1.start();
        // Exception as the thread is already started
        t1.setDaemon(true);
        t2.start();
    }
}

20190917003508913.png


例子

package com.artisan.test;
import java.time.LocalDateTime;
public class DaemonThread extends Thread {
    public DaemonThread(String name) {
        super(name);
    }
    @Override
    public void run() {
        // Checking whether the thread is Daemon or not
        if (Thread.currentThread().isDaemon()) {
            try {
                System.out.println(getName() + " is Daemon thread : running " + LocalDateTime.now());
                // 休眠200s
                Thread.sleep(200_000);
                System.out.println(getName() + " is Daemon thread: over " + LocalDateTime.now());
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        } else {
            try {
                System.out.println(getName() + " is User thread : running " + LocalDateTime.now());
                // 休眠5s
                Thread.sleep(5_000);
                System.out.println(getName() + " is User thread : over " + LocalDateTime.now());
            } catch (InterruptedException e) {
                e.printStackTrace();
            }
        }
    }
    public static void main(String[] args) {
        System.out.println(Thread.currentThread().getName() + ": running " + LocalDateTime.now());
        DaemonThread t1 = new DaemonThread("t1");
        DaemonThread t2 = new DaemonThread("t2");
        DaemonThread t3 = new DaemonThread("t3");
        // Setting user thread t1 to Daemon
        t1.setDaemon(true);
        // starting first 2 threads
        t1.start();
        t2.start();
        // Setting user thread t3 to Daemon
        t3.setDaemon(true);
        t3.start();
        System.out.println(Thread.currentThread().getName() + ": over " + LocalDateTime.now());
    }
}


执行结果:


20190917002248927.png


setDaemon(true) 设置为Daemon Thread

JVM将在终止任务之前等待任何用户线程完成其任务,JVM不关心守护程序线程是否正在运行,当用户线程结束后将退出。 从日志中我们可以看到t2是个user thread ,休眠了5秒,t3是daemon thread 休眠200秒,但是我们看到t2 用户线程执行完成后,jvm就退出了,虽然t3 daemon thread 还在进行中,这个时候t3已经被终止了。


使用场景分析


心跳检测


A ----------------------------------------------------------------------------- B

–>Daemon Thread(Health Check)


举个例子: 当A到B建立了一个长连接 ,长连接是需要发心跳的,维持这个连接。 这个时候可以在中开启一个Daemon Thread用于心跳检测,当A死掉的时候,这个Daemon Thread 也会被JVM终止掉,就避免了A和B之间已经断开,但是心跳检测可能报错了但一直不退出的情况的发生。

相关文章
|
7月前
|
存储 Java
高并发编程之多线程锁和Callable&Future 接口
高并发编程之多线程锁和Callable&Future 接口
84 1
|
7月前
|
缓存 监控 Java
高并发编程之ThreadPool 线程池
高并发编程之ThreadPool 线程池
78 1
|
2月前
|
并行计算 算法 搜索推荐
探索Go语言的高并发编程与性能优化
【10月更文挑战第10天】探索Go语言的高并发编程与性能优化
|
3月前
|
网络协议 Java Linux
高并发编程必备知识IO多路复用技术select,poll讲解
高并发编程必备知识IO多路复用技术select,poll讲解
|
2月前
|
Java Linux 应用服务中间件
【编程进阶知识】高并发场景下Bio与Nio的比较及原理示意图
本文介绍了在Linux系统上使用Tomcat部署Java应用程序时,BIO(阻塞I/O)和NIO(非阻塞I/O)在网络编程中的实现和性能差异。BIO采用传统的线程模型,每个连接请求都会创建一个新线程进行处理,导致在高并发场景下存在严重的性能瓶颈,如阻塞等待和线程创建开销大等问题。而NIO则通过事件驱动机制,利用事件注册、事件轮询器和事件通知,实现了更高效的连接管理和数据传输,避免了阻塞和多级数据复制,显著提升了系统的并发处理能力。
66 0
|
7月前
|
存储 关系型数据库 MySQL
《MySQL 入门教程》第 05 篇 账户和权限,Java高并发编程详解深入理解pdf
《MySQL 入门教程》第 05 篇 账户和权限,Java高并发编程详解深入理解pdf
|
7月前
|
Java
高并发编程之JUC 三大辅助类和读写锁
高并发编程之JUC 三大辅助类和读写锁
54 1
|
7月前
|
消息中间件 Java Linux
2024年最全BATJ真题突击:Java基础+JVM+分布式高并发+网络编程+Linux(1),2024年最新意外的惊喜
2024年最全BATJ真题突击:Java基础+JVM+分布式高并发+网络编程+Linux(1),2024年最新意外的惊喜
|
6月前
|
缓存 NoSQL Java
Java高并发实战:利用线程池和Redis实现高效数据入库
Java高并发实战:利用线程池和Redis实现高效数据入库
514 0
|
4月前
|
监控 算法 Java
企业应用面临高并发等挑战,优化Java后台系统性能至关重要
随着互联网技术的发展,企业应用面临高并发等挑战,优化Java后台系统性能至关重要。本文提供三大技巧:1)优化JVM,如选用合适版本(如OpenJDK 11)、调整参数(如使用G1垃圾收集器)及监控性能;2)优化代码与算法,减少对象创建、合理使用集合及采用高效算法(如快速排序);3)数据库优化,包括索引、查询及分页策略改进,全面提升系统效能。
52 0