详解Java如何关闭线程以及线程池

 更新时间:2022年04月18日 10:22:31   作者:码农研究僧  
java如何正确关闭线程以及线程池是一个高频的面试题,本文将为大家详细介绍实现的方法与代码,感兴趣的小伙伴快跟随小编一起学习一下

前言

这个问题是一个高频的面试题

而且在印象中是由stop方法执行或者终端中的kill杀死

但是这些方法直接简单粗暴,很不安全,而且也不推广

不使用stop的方法

之所以不安全不推广是因为:

  • stop方法不管线程逻辑是否完整,都会终止当前正在运行的线程
  • 会破坏其原子逻辑(多线程加了锁之解决资源共享,但是stop会将其所有锁丢弃,造成混乱)

1. 关闭线程

1.1 volatile关键字

使用自定义的标志位决定线程的执行情况

具体思路大致如下:设置一个 父线程 的状态变量,以其影响其子线程即可

public class test extends Thread {
    //标识线程是否结束
    public static boolean thread_stop = true;

    public void stopThread() {
        thread_stop = false;
    }


    public static void main(String[] args) {
        test t = new test();
        t.start();
        System.out.println("Father Thread Start");
        try {
            //先让线程跑起来
            Thread.sleep(2);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        //结束线程

        //将其状态变量直接改为false
        //thread_stop = false;

        //调用方法改为false(与状态变量直接修改 一个道理)
        t.stopThread();

        System.out.println("Father Thread end");
    }


    @Override
    public void run() {
        while (thread_stop) {
            System.out.println("Child Thread Start");
        }
        System.out.println("Child Thread end");
    }
}

但是网上说不加volatile是停不下来的,其实是可以停下来的

只不过

加了volatile有几个好处:

  • volatile可以保证状态变量为系统内存值而不是缓存里值(避免值不一致)
  • volatile 关键字能够是该变量对其他线程“可见”,即当主线程修改变量并刷新到主内存后,会让其他线程去主内存中读取该变量。当然,volatile并不能保证线程的安全性

具体加在状态变量中的位置如下:

//标识线程是否结束
public static volatile boolean thread_stop = true;

之后具体完整的输出为:

具体完整的输出为:

Father Thread start
Child Thread Start
Child Thread Start
。。。
。。。
Child Thread Start
Child Thread Start
Father Thread end
Child Thread end

1.2 intrrrupt()方法

不能终止一个正在执行着的线程,它只是修改中断标志而已

这个方法分为两种情况:

线程处于阻塞:立马退出阻塞,抛出InterruptedException异常。通过捕获这个异常,来让线程退出

线程处于非阻塞:处于运行状态不受影响,仅仅标记了线程的中断为true。在适当的位置中调用isInterrupted方法查看是否被中断并且退出

public class test extends Thread {
    public static void main(String[] args) {
        test t = new test();
        t.start();
        System.out.println("Father Thread Start");
        try {
            //先让线程跑起来
            Thread.sleep(2);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        //结束线程

        t.interrupt();

        System.out.println("Father Thread end");
    }


    @Override
    public void run() {
    //分配线程的中断状态,并且此状态可以由interrupted()方法生成
        while (!Thread.interrupted()) {
            System.out.println("Child Thread Start");
        }
        System.out.println("Child Thread end");
    }
}

执行结果截图:

具体完整的输出为:

Father Thread start
Child Thread Start
Child Thread Start
。。。
。。。
Child Thread Start
Child Thread Start
Father Thread end
Child Thread end

2.关闭线程池

优雅的关闭线程池:(比如ThreadPoolExecutor类)

可以通过shutdown方法逐步关闭池中的线程(温和安全)

shutdown():拒收新任务,不会立即终止线程池。而是要等所有任务缓存队列中的任务都执行完后才终止。

shutdownNow():拒收新任务,立即终止线程池。并尝试打断正在执行的任务。

并且清空任务缓存队列,返回尚未执行的任务

以下是对两个线程池关闭的方法源代码进行分析

而且关闭的途中,这两个方法也不是瞬间立马关闭,等待关闭的同时,还还调用awaitTermination方法来阻塞等待

2.1 shutdownNow()方法

查看java的源代码

在try内部结构中

1.检查其状态

2.原子性的修改线程池的状态为stop

3.遍历工作队列线程,调用interrupt方法

4.将队列中还未执行的放到任务队列

源码内部:其逻辑就是修改线程池状态为stop,工作队列中调用interrupt方法

在调用shutdownNow方法:

  • 正在执行的线程会(getTask返回null)导致线程退出。
  • 队列中读取的任务会阻塞,抛出异常之后。工作队列就会调用interrupt方法

2.2 shutdown()方法

同样也是看java的源代码

同样也是4步状态

1.检查其状态

2.修改线程池状态为SHUTDOWN

3.调用interruptIdleWorkers方法中断空闲线程(只有加锁成功的线程才会被调用interrupt方法)

而正在执行的线程是加锁失败,不会被中断

主要通过这个函数去区分判断

以上就是详解Java如何关闭线程以及线程池的详细内容,更多关于Java关闭线程 线程池的资料请关注脚本之家其它相关文章!

相关文章

  • springboot启动不加载bootstrap.yml文件的问题

    springboot启动不加载bootstrap.yml文件的问题

    这篇文章主要介绍了springboot启动不加载bootstrap.yml文件的问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-12-12
  • Java FileDescriptor总结_动力节点Java学院整理

    Java FileDescriptor总结_动力节点Java学院整理

    FileDescriptor 是“文件描述符”。可以被用来表示开放文件、开放套接字等。接下来通过本文给大家分享Java FileDescriptor总结,感兴趣的朋友一起学习吧
    2017-05-05
  • Java全面分析面向对象之继承

    Java全面分析面向对象之继承

    继承就是可以直接使用前辈的属性和方法。自然界如果没有继承,那一切都是处于混沌状态。多态是同一个行为具有多个不同表现形式或形态的能力。多态就是同一个接口,使用不同的实例而执行不同操作
    2022-04-04
  • SpringMVC实现RESTful风格:@PathVariable注解的使用方式

    SpringMVC实现RESTful风格:@PathVariable注解的使用方式

    这篇文章主要介绍了SpringMVC实现RESTful风格:@PathVariable注解的使用方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2021-11-11
  • java如何利用NIO压缩文件或文件夹

    java如何利用NIO压缩文件或文件夹

    这篇文章主要介绍了java如何利用NIO压缩文件或文件夹问题,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2023-12-12
  • 利用javaFX实现移动一个小球的示例代码

    利用javaFX实现移动一个小球的示例代码

    这篇文章主要介绍了利用javaFX实现移动一个小球的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-09-09
  • IDEA中编写并运行shell脚本的实现

    IDEA中编写并运行shell脚本的实现

    这篇文章主要介绍了IDEA中编写并运行shell脚本的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2020-08-08
  • java中接口和事件监听器的深入理解

    java中接口和事件监听器的深入理解

    这篇文章主要给大家介绍了关于java中接口和事件监听器的相关资料,文中通过示例代码介绍的非常详细,对大家学习或者使用java具有一定的参考学习价值,需要的朋友们下面来一起学习学习吧
    2019-12-12
  • Java使用jni清屏功能的实现(只针对cmd)

    Java使用jni清屏功能的实现(只针对cmd)

    JNI是Java Native Interface的缩写,它提供了若干的API实现了Java和其他语言的通信(主要是C&C++)。这篇文章主要介绍了Java使用jni清屏功能的实现(只针对cmd) ,感兴趣的朋友跟随脚本之家小编一起学习吧
    2018-05-05
  • 详解Spring Bean的集合注入和自动装配

    详解Spring Bean的集合注入和自动装配

    这篇文章主要为大家详细介绍了Spring Bean中集合注入和自动装配的方法,文中的示例代码讲解详细,对我们学习有一定的帮助,需要的可以参考一下
    2022-06-06

最新评论