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


在这里插入图片描述

在这里插入图片描述

官方文档

我们以JAVA8的doc为例 戳这里

在这里插入图片描述


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


在这里插入图片描述


例子

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


执行结果:

在这里插入图片描述

  1. setDaemon(true) 设置为Daemon Thread
  2. 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之间已经断开,但是心跳检测可能报错了但一直不退出的情况的发生。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

小小工匠

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值