Java中如何安全地停止线程?
大家好,我是锋哥。今天分享关于【Java中如何安全地停止线程?】面试题。希望对大家有帮助;
Java中如何安全地停止线程?
1000道 互联网大厂Java工程师 精选面试题-Java资源分享网
在Java中,安全地停止线程是一项重要的任务,尤其是在并发编程中。线程如果被强制停止,可能会导致资源泄漏、数据不一致等问题。因此,我们必须确保线程停止的方式既要有效,又要保证线程执行过程中的资源得到正确的释放。
Java中提供了多种方式来安全地停止线程,主要有以下几种:
1. 使用 volatile
标志位
使用一个共享的 volatile
变量来标记线程是否应当停止。volatile
关键字确保变量的变化能够被其他线程及时看到,避免了线程间的缓存问题。
示例代码:
public class SafeThreadStop implements Runnable {private volatile boolean running = true; // 标志位@Overridepublic void run() {while (running) {// 线程执行的任务System.out.println("Thread is running...");try {Thread.sleep(1000); // 模拟一些工作} catch (InterruptedException e) {Thread.currentThread().interrupt(); // 设置中断标志}}System.out.println("Thread has stopped safely.");}// 停止线程的方法public void stopThread() {running = false; // 修改标志位,线程将退出循环}public static void main(String[] args) throws InterruptedException {SafeThreadStop safeThreadStop = new SafeThreadStop();Thread thread = new Thread(safeThreadStop);thread.start();// 等待一段时间后停止线程Thread.sleep(5000);safeThreadStop.stopThread();}
}
解释:
- 使用
volatile boolean running
作为停止线程的标志位。 - 在
run()
方法的while (running)
循环中检查标志位,决定是否继续执行。 - 通过调用
stopThread()
方法,将标志位设置为false
,使得线程退出循环,最终安全停止。
优点:
- 简单、直观。
- 不会强制中断线程,允许线程在适当的位置检查并自行终止。
缺点:
- 如果线程正在执行长时间运行的任务,它可能不会立刻停止,必须通过合适的检查条件来确保线程能够及时退出。
2. 使用 Thread.interrupt()
方法
Thread.interrupt()
是一个用于中断线程的常见方法,但需要线程在合适的地方主动响应中断请求。通过捕获 InterruptedException
异常或定期检查线程的中断状态,线程可以安全地停止。
示例代码:
public class InterruptThreadStop implements Runnable {@Overridepublic void run() {while (!Thread.currentThread().isInterrupted()) { // 检查中断状态// 线程执行的任务System.out.println("Thread is running...");try {Thread.sleep(1000); // 模拟一些工作} catch (InterruptedException e) {// 当sleep方法被中断时,恢复中断标志Thread.currentThread().interrupt();break; // 中断后安全退出线程}}System.out.println("Thread has stopped safely.");}public static void main(String[] args) throws InterruptedException {InterruptThreadStop interruptThreadStop = new InterruptThreadStop();Thread thread = new Thread(interruptThreadStop);thread.start();// 等待一段时间后中断线程Thread.sleep(5000);thread.interrupt(); // 中断线程}
}
解释:
- 在线程的
run()
方法中,使用Thread.currentThread().isInterrupted()
来检查线程的中断状态。 - 如果线程被中断,
InterruptedException
会被抛出,可以在异常处理块中恢复中断标志并跳出循环,安全地退出线程。
优点:
Thread.interrupt()
是一个非强制性方法,它不会强制停止线程,而是通过让线程自己检查中断标志来实现停止。- 可以优雅地响应中断,使线程能够在合适的时机停下来。
缺点:
- 必须在线程的执行过程中主动检查中断标志或捕获
InterruptedException
异常,线程才会在中断时停止。
3. 使用 ExecutorService
的 shutdown()
或 shutdownNow()
方法
对于通过线程池(ExecutorService
)管理的线程,使用 shutdown()
或 shutdownNow()
方法来停止线程池中的线程是推荐的做法。这些方法能够通过协调线程池的状态来安全地停止线程。
shutdown()
:平滑关闭,线程池会完成已经提交的任务,但不会接受新的任务。shutdownNow()
:立即关闭,尝试停止所有正在执行的任务,并返回尚未开始的任务列表。
示例代码:
import java.util.concurrent.*;public class ExecutorServiceStop {public static void main(String[] args) throws InterruptedException {ExecutorService executorService = Executors.newFixedThreadPool(2);Runnable task = () -> {while (!Thread.currentThread().isInterrupted()) {// 执行任务System.out.println("Thread is running...");try {Thread.sleep(1000);} catch (InterruptedException e) {Thread.currentThread().interrupt(); // 处理中断}}System.out.println("Thread has stopped safely.");};executorService.submit(task);executorService.submit(task);// 等待一段时间后停止线程池Thread.sleep(5000);executorService.shutdown(); // 停止线程池,不能再接受新任务}
}
解释:
ExecutorService
提供了更高层次的线程管理,能够控制线程的启动、停止以及任务的提交。- 调用
shutdown()
后,线程池将停止接受新任务,但会继续执行已提交的任务。 - 调用
shutdownNow()
会立即尝试停止所有正在执行的任务,并返回未开始的任务。
优点:
- 使用
ExecutorService
管理线程池时,提供了更方便和安全的线程停止机制。 - 线程池在应用程序中可以很方便地管理线程的生命周期。
缺点:
- 需要线程池来管理线程,因此需要对线程池进行配置和管理。
4. 使用 Future.cancel()
方法
如果线程任务是通过 ExecutorService.submit()
提交的,可以通过 Future
对象的 cancel()
方法来尝试取消正在执行的任务。
示例代码:
import java.util.concurrent.*;public class FutureCancelStop {public static void main(String[] args) throws InterruptedException {ExecutorService executorService = Executors.newFixedThreadPool(1);Runnable task = () -> {while (!Thread.currentThread().isInterrupted()) {// 执行任务System.out.println("Thread is running...");try {Thread.sleep(1000);} catch (InterruptedException e) {Thread.currentThread().interrupt(); // 处理中断}}System.out.println("Thread has stopped safely.");};Future<?> future = executorService.submit(task);// 等待一段时间后取消任务Thread.sleep(5000);future.cancel(true); // 取消任务,并尝试中断正在执行的线程executorService.shutdown();}
}
解释:
cancel(true)
尝试取消正在执行的任务并中断线程。需要线程本身响应中断(如在sleep
或wait
等方法上处理中断)。
总结
安全地停止线程的方法有多种,关键是确保线程在停止前能够释放资源并完成必要的清理工作。常见的线程停止方式包括:
- 使用
volatile
标志位:适合任务具有周期性检查条件的场景。 - 使用
Thread.interrupt()
:通过中断线程,要求线程在合适的地方响应中断并退出。 - 使用
ExecutorService
的shutdown()
或shutdownNow()
:通过线程池管理线程的生命周期,平滑停止线程。 - 使用
Future.cancel()
:通过Future
对象尝试取消任务并中断线程。
以上方法都可以在不同场景中确保线程以一种安全、优雅的方式停止。
相关文章:

Java中如何安全地停止线程?
大家好,我是锋哥。今天分享关于【Java中如何安全地停止线程?】面试题。希望对大家有帮助; Java中如何安全地停止线程? 1000道 互联网大厂Java工程师 精选面试题-Java资源分享网 在Java中,安全地停止线程是一项重要的任务,尤其…...

Apache Tomcat文件包含漏洞复现(详细教程)
1.漏洞原理 Tomcat 服务器是一个免费的开放源代码的Web 应用服务器,其安装后会默认开启ajp连接器,方便与其他web服务器通过ajp协议进行交互。属于轻量级应用服务器,在中小型系统和并发访问用户不是很多的场合下被普遍使用,是开发…...

个人学习 - 什么是Vim?
观我往旧,同我仰春 - 2025.1.10 声明 仅作为个人学习使用,仅供参考 本文所有解释参考笔者个人理解,最终目的是服务于自我学习, 如果你需要了解官方更规范的解释,请自行查阅 Vim 是什么 Vim 是一个强大的 文本编辑器…...

Flink Gauss CDC:深度剖析存量与增量同步的创新设计
目录 设计思路 1.为什么不直接用FlinkCDC要重写Flink Gauss CDC 2.存量同步的逻辑是什么 2.1、单主键的切片策略是什么 2.2、复合主键作切片,怎么保证扫描到所有的数据 3、增量同步的逻辑是什么 4、存量同步结束之后如何无缝衔接增量同步 5、下游数据如何落…...
docker 部署.netcore应用优势在什么地方?
目录 1. 环境一致性 2. 简化依赖管理 3. 快速部署与扩展 4. 资源利用率高 5. 版本控制与回滚 6. 安全性 7. 生态系统支持 8. 微服务架构支持 9. 降低成本 10. 开发体验提升 总结 使用 Docker 部署 .NET Core 应用有许多优势,特别是在开发、测试和生产环境…...
AIP-126 枚举
编号126原文链接AIP-126: Enumerations状态批准创建日期2019-07-24更新日期2019-07-24 一个域的值集合是一组数量有限的具体值,这是很常见的。此时使用枚举(缩写为“enums”)可有助于明确表达值集合的范围。 指南 API 可以 为不经常更改的…...
P3707 [SDOI2017] 相关分析 Solution
Description 给定序列 x ( x 1 , x 2 , ⋯ , x n ) , y ( y 1 , y 2 , ⋯ , y n ) x(x_1,x_2,\cdots,x_n),y(y_1,y_2,\cdots,y_n) x(x1,x2,⋯,xn),y(y1,y2,⋯,yn),有 m m m 个操作,分三种: query ( l , r ) \operatornam…...

Android AutoMotive --CarService
1、AAOS概述 Android AutoMotive OS是谷歌针对车机使用场景打造的操作系统,它是基于现有Android系统的基础上增加了新特性,最主要的就是增加了CarService(汽车服务)模块。我们很容易把Android AutoMotive和Android Auto搞混&…...

K8S中Service详解(三)
HeadLiness类型的Service 在某些场景中,开发人员可能不想使用Service提供的负载均衡功能,而希望自己来控制负载均衡策略,针对这种情况,kubernetes提供了HeadLiness Service,这类Service不会分配Cluster IP,…...

C++----STL(vector)
vector的介绍 vector的文档介绍:cplusplus.com/reference/vector/vector/ 1.基本概念 简单来说,vector是表示可以改变大小的数组的顺序容器。使用连续的存储位置来存储元素,因此可以通过常规指针的偏移量来高效访问。 2.内部机制 vector…...

Ubuntu24.04初始化MySQL报错 error while loading shared libraries libaio.so.1
Ubuntu24.04初始化MySQL报错 error while loading shared libraries: libaio.so.1 问题一:libaio1不存在 # 提示libaio1不存在 [rootzabbix-mysql-master.example.com x86_64-linux-gnu]#apt install numactl libaio1 Reading package lists... Done Building depe…...

初探大数据流式处理
文章目录 初探大数据流式处理批式处理系统特点流式处理系统特点大批次计算微批次计算适用场景 流式计算的应用场景流式大数据的特征流式计算的关键技术流式处理框架的特征三大流式数据处理框架 初探大数据流式处理 大数据处理系统主要分为批式处理和流式处理两类。批式处理将大…...

【Linux】Linux入门(三)权限
目录 前提权限概念whoami指令 Linux权限管理文件访问者的分类(人)file指令权限信息权限的表示方法 chmod指令 更改权限chown指令 修改文件,文件夹所属用户和用户组 权限掩码umask(权限掩码) 粘滞位 前提 请先看下面这…...

html,css,js的粒子效果
这段代码实现了一个基于HTML5 Canvas的高级粒子效果,用户可以通过鼠标与粒子进行交互。下面是对代码的详细解析: HTML部分 使用<!DOCTYPE html>声明文档类型。<html>标签内包含了整个网页的内容。<head>部分定义了网页的标题&#x…...
Spring Boot + Netty + WebSocket 实现消息推送
1、关于Netty Netty 是一个利用 Java 的高级网络的能力,隐藏其背后的复杂性而提供一个易于使用的 API 的客户端/服务器框架。 2、Maven依赖 <dependencies><!-- https://mvnrepository.com/artifact/io.netty/netty-all --><dependency><gr…...
Python3 【字符串】:方法和函数使用示例手册
Python3 【字符串】:方法和函数使用示例手册 Python 提供了丰富的字符串处理方法和函数,以下是一些常用的方法和函数分类整理,并提供详细使用示例,简单易懂,值得收藏。 1. 字符串大小写转换 str.upper():…...

数据结构与算法整理复习(一):数据结构概念与线性表
目录 第一章:绪论 1.1 数据结构的基本概念 1.2 算法与算法评价 第二章:线性表 2.1 线性表的定义和基本操作 2.2 线性表的顺序表示(顺序表) 应用题 2.3 线性表的链式表达(链表) 2.3.1 单链表 2.3.2…...

【Block总结】PConv风车卷积,更大的感受野,提高特征提取能力|即插即用
论文信息 论文标题:《Pinwheel-shaped Convolution and Scale-based Dynamic Loss for Infrared Small Target Detection》 论文链接:https://arxiv.org/pdf/2412.16986 GitHub链接:https://github.com/JN-Yang/PConv-SDloss-Data 创新点 …...

Python新春烟花
目录 系列文章 写在前面 技术需求 完整代码 下载代码 代码分析 1. 程序初始化与显示设置 2. 烟花类 (Firework) 3. 粒子类 (Particle) 4. 痕迹类 (Trail) 5. 烟花更新与显示 6. 主函数 (fire) 7. 游戏循环 8. 总结 注意事项 写在后面 系列文章 序号直达链接爱…...

VirtualBox can‘t enable the AMD-V extension
个人博客地址:VirtualBox cant enable the AMD-V extension | 一张假钞的真实世界 最近一次完成Deepin的系统更新后,进入VirtualBox创建的虚拟机(Widows10)时,出现以下错误: 根据网址“https://askubuntu.…...

TDengine 快速体验(Docker 镜像方式)
简介 TDengine 可以通过安装包、Docker 镜像 及云服务快速体验 TDengine 的功能,本节首先介绍如何通过 Docker 快速体验 TDengine,然后介绍如何在 Docker 环境下体验 TDengine 的写入和查询功能。如果你不熟悉 Docker,请使用 安装包的方式快…...

在WSL2的Ubuntu镜像中安装Docker
Docker官网链接: https://docs.docker.com/engine/install/ubuntu/ 1、运行以下命令卸载所有冲突的软件包: for pkg in docker.io docker-doc docker-compose docker-compose-v2 podman-docker containerd runc; do sudo apt-get remove $pkg; done2、设置Docker…...

Yolov8 目标检测蒸馏学习记录
yolov8系列模型蒸馏基本流程,代码下载:这里本人提交了一个demo:djdll/Yolov8_Distillation: Yolov8轻量化_蒸馏代码实现 在轻量化模型设计中,**知识蒸馏(Knowledge Distillation)**被广泛应用,作为提升模型…...
NPOI操作EXCEL文件 ——CAD C# 二次开发
缺点:dll.版本容易加载错误。CAD加载插件时,没有加载所有类库。插件运行过程中用到某个类库,会从CAD的安装目录找,找不到就报错了。 【方案2】让CAD在加载过程中把类库加载到内存 【方案3】是发现缺少了哪个库,就用插件程序加载进…...
MySQL 索引底层结构揭秘:B-Tree 与 B+Tree 的区别与应用
文章目录 一、背景知识:什么是 B-Tree 和 BTree? B-Tree(平衡多路查找树) BTree(B-Tree 的变种) 二、结构对比:一张图看懂 三、为什么 MySQL InnoDB 选择 BTree? 1. 范围查询更快 2…...
跨平台商品数据接口的标准化与规范化发展路径:淘宝京东拼多多的最新实践
在电商行业蓬勃发展的当下,多平台运营已成为众多商家的必然选择。然而,不同电商平台在商品数据接口方面存在差异,导致商家在跨平台运营时面临诸多挑战,如数据对接困难、运营效率低下、用户体验不一致等。跨平台商品数据接口的标准…...
写一个shell脚本,把局域网内,把能ping通的IP和不能ping通的IP分类,并保存到两个文本文件里
写一个shell脚本,把局域网内,把能ping通的IP和不能ping通的IP分类,并保存到两个文本文件里 脚本1 #!/bin/bash #定义变量 ip10.1.1 #循环去ping主机的IP for ((i1;i<10;i)) doping -c1 $ip.$i &>/dev/null[ $? -eq 0 ] &&am…...
算法250609 高精度
加法 #include<stdio.h> #include<iostream> #include<string.h> #include<math.h> #include<algorithm> using namespace std; char input1[205]; char input2[205]; int main(){while(scanf("%s%s",input1,input2)!EOF){int a[205]…...
FTXUI::Dom 模块
DOM 模块定义了分层的 FTXUI::Element 树,可用于构建复杂的终端界面,支持响应终端尺寸变化。 namespace ftxui {...// 定义文档 定义布局盒子 Element document vbox({// 设置文本 设置加粗 设置文本颜色text("The window") | bold | color(…...
第22节 Node.js JXcore 打包
Node.js是一个开放源代码、跨平台的、用于服务器端和网络应用的运行环境。 JXcore是一个支持多线程的 Node.js 发行版本,基本不需要对你现有的代码做任何改动就可以直接线程安全地以多线程运行。 本文主要介绍JXcore的打包功能。 JXcore 安装 下载JXcore安装包&a…...