多线程之interrupt与优雅停止一个线程
1.背景
在实际开发中,我们可能会遇到终止某个线程的场景,
比如不断扫描数据库的发货订单时,这时候需停止扫描,
当然我们不能把程序关了,我们只希望停止扫描数据库这一个线程,
那么应该怎么办了?
这就可以使用线程中提供的interrupt()这个方法
2.案例演示
package com.ldp.demo01; import com.common.MyThreadUtil;
import lombok.extern.slf4j.Slf4j;
import org.junit.Test; /**
* @author 姿势帝-博客园
* @address https://www.cnblogs.com/newAndHui/
* @WeChat 851298348
* @create 01/16 9:41
* @description <p>
* interrupt()方法的理解
* 1.只是给线程设置一个打断标记,不会改变线程的实际状态
* 2.打断睡眠中的线程会清除打断标记(即标记无效)
* 3.实际应用中可以利用判断线程的打断标记,来判断是否继续业务.
* </p>
*/
@Slf4j
public class Test03Interrupt {
/**
* 打断正常的线程
* 1. t1.interrupt() 只是给线程设置一个打断标记,不会改变线程的实际状态
*/
@Test
public void test01() {
Thread t1 = new Thread(() -> {
log.info("执行中......");
while (true) {
}
}, "t1");
// 启动线程
t1.start();
// 查看打断状态
log.info("interrupted=" + t1.isInterrupted());
// 查看线程状态
log.info("线程状态为:" + t1.getState());
// 睡眠2秒
MyThreadUtil.sleep(2);
// 打断线程
t1.interrupt();
// 查看打断状态
log.info("interrupted=" + t1.isInterrupted());
// 再次查看线程状态
log.info("线程状态为:" + t1.getState());
// 防止主线程结束
MyThreadUtil.sleep(10);
} /**
* 打断处于睡眠的线程
* 1. 打断睡眠中的线程会清除打断标记(即标记无效)
*/
@Test
public void test02() {
Thread t1 = new Thread(() -> {
while (true) {
log.info("执行中......");
Thread currentThread = Thread.currentThread();
if (currentThread.isInterrupted()) {
log.info("isInterrupted=true");
break;
}
try {
Thread.sleep(50 * 1000);
} catch (InterruptedException e) {
e.printStackTrace();
log.info("isInterrupted=" + currentThread.isInterrupted());
// 重新设置打断标记
currentThread.interrupt();
}
}
}, "t1");
// 启动线程
t1.start();
// 睡眠2秒
MyThreadUtil.sleep(2);
// 打断线程
t1.interrupt();
// 查看打断状态
log.info("isInterrupted=>" + t1.isInterrupted());
// 防止主线程结束
MyThreadUtil.sleep(10);
} /**
* 两阶段终止
* 案例:
* 假设有一个主线程一直在扫描数据的订单进行发货操作,
* 在页面有我们需要有暂停发货\继续发货\停止发货三个功能
*/
@Test
public void test03() {
// 扫描数据线程
Thread threadDataMain = new Thread(() -> {
Thread currentThread = Thread.currentThread();
while (true) {
boolean interrupted = currentThread.isInterrupted();
if (interrupted) {
// 如果线程被打断就停止循环
log.info("停止获取数据");
break;
}
// 模拟读取数据数据,每次1条
String order = getOrder();
if (order == null) {
log.info("无数据休息2秒");
// 数据库无订单休息2秒
try {
Thread.sleep(2 * 1000);
} catch (InterruptedException e) {
e.printStackTrace();
// 重新设置打断标记
currentThread.interrupt();
}
continue;
}
// 发货线程
Thread threadOrder = new Thread(() -> {
log.info("订单发货中:" + order);
}, "t-" + order);
threadOrder.start();
}
}, "thread-数据库扫描主线程");
threadDataMain.start(); // 模拟发货10秒后停止发货
MyThreadUtil.sleep(10);
threadDataMain.interrupt(); // 防止主线程结束
MyThreadUtil.sleep(20);
} /**
* 模拟数据库获取订单
*
* @return
*/
public String getOrder() {
// 模拟有时候无数据的情况
// int nextInt = new Random().nextInt(10);
long millis = System.currentTimeMillis();
if (millis % 3 > 0) {
return null;
}
// 3的整数倍才返回订单
return "NO" + System.currentTimeMillis();
}
}
完美!
多线程之interrupt与优雅停止一个线程的更多相关文章
- Java线程状态、线程start方法源码、多线程、Java线程池、如何停止一个线程
下面将依次介绍: 1. 线程状态.Java线程状态和线程池状态 2. start方法源码 3. 什么是线程池? 4. 线程池的工作原理和使用线程池的好处 5. ThreadPoolExecutor中的 ...
- Java 如何正确停止一个线程
自己在做实验性小项目的时候,发现自己遇到一个问题:如何控制线程的"死亡"? 首先,如何开启一个线程呢? 最简单的代码: public class Main { public sta ...
- Java并发(基础知识)—— 创建、运行以及停止一个线程
在计算机世界,当人们谈到并发时,它的意思是一系列的任务在计算机中同时执行.如果计算机有多个处理器或者多核处理器,那么这个同时性是真实发生的:如果计算机只有一个核心处理器那么就只是表面现象. 现代所有的 ...
- Java多线程之interrupt()的深度研究
近期学习Java多线程的中断机制,网上的帖子说得很浅,并没深究其原理.看了Java源码,对Java的中断机制有了略深入的理解,在这篇文章中向感兴趣的网友分享下.这篇文章主要通过一个典型例子对中断机制进 ...
- java如何正确停止一个线程
Thread类中有start(), stop()方法,不过stop方法已经被废弃掉. 平时其实也有用过,共享一个变量,相当于标志,不断检查标志,判断是否退出线程 如果有阻塞,需要使用Thread的in ...
- java停止一个线程
Thread类中有start(), stop()方法,不过stop方法已经被废弃掉. 平时其实也有用过,共享一个变量,相当于标志,不断检查标志,判断是否退出线程 如果有阻塞,需要使用Thread的in ...
- java基础---->多线程之interrupt(九)
这里我们通过实例来学习一下java多线程中关于interrupt方法的一些知识.执者失之.我想当一个诗人的时候,我就失去了诗,我想当一个人的时候,我就失去了我自己.在你什么也不想要的时候,一切如期而来 ...
- java中的线程(2):如何正确停止线程之3种常见停止方式
1.常见停止方式 自定义线程,其中含退出标志位,在run中判断它. 使用interrupt()方法中断线程 使用stop方法暴力终止(已经弃用) 2.使用标志位 class TestThread ex ...
- java中的线程(2):如何正确停止线程之2种常见停止方式
1.常见停止方式 结束run函数,run中含退出标志位. 使用interrupt()方法中断线程 使用stop方法暴力终止(已经弃用) 2.结束run class TestThread extends ...
- 多线程之interrupt
1.interrupt()作为中断程序,并不会直接终止运行,而是设置中断状态,由线程自己处理中断.可以选择终止线程.等待新任务或继续执行. 2.interrupt()经常用于中断处于堵塞状态的的线程, ...
随机推荐
- ESM风潮下企业服务的最佳实践探讨
甄知科技孵化于中国领先的IT咨询服务提供商-上海汉得信息技术股份有限公司,主打产品"燕千云"于2019年正式发布,持续迭代版本至今,燕千云作为企业数字化服务平台,燕千云的愿景和现状 ...
- Flink状态(二)
Flink提供了不同的状态存储方式,并说明了状态如何存和存储在哪里. 状态可以被存储在Jvm的堆和堆外.根据状态存储方式的不同,Flink也能代替应用管理状态,意思是Flink能够进行内存管理(有必要 ...
- 开源云同步的markdown写作软件——Yosoro
文章目录 前言 简便的项目管理 舒服的写作体验 支持one driver 存在缺点 前言 Yosoro是一款支持在Win.Linux.macOS上使用的写作软件.它的界面设计以及交互上表达出的极简主义 ...
- NVIDIA Jetson AGX Xavier 从刷机之后到配置环境
特殊的配置环境需求: cuda-10.2.python 3.6.9.torch 1.7.0.torchversion 0.8.1,剩下的顺其自然即可(逃. 顺便说一句,里面的指令请一行一行仔细复制粘贴 ...
- 【译】Visual Studio 2022 - 17.10 性能增强
我们很高兴地宣布 Visual Studio 2022 的最新更新,它为您带来了 IDE 各个领域的一系列性能增强.在这篇博客中,我们将重点介绍17.10版本中一些最显著的改进,比如更快的 Windo ...
- Web运作原理探析
Web运作原理探析 1.1 web的 概念 Web是一种分布式的应用架构,旨在共享分布在网络上的各个Web服务器中的所有互相链接的信息. 1.2 HTML是指超文本标记语言. 1.3 URL简介 UR ...
- adb连接安卓设备失败failed to start daemon
adb连接安卓设备失败failed to start daemon Reference:https://blog.csdn.net/whshuo2010/article/details/5109449 ...
- hypernetwork在SD中是怎么工作的
大家在stable diffusion webUI中可能看到过hypernetwork这个词,那么hypernetwork到底是做什么用的呢? 简单点说,hypernetwork模型是用于修改样式的小 ...
- position的值, relative和absolute分别是相对于谁进行定位的?
relative: 相对定位,相对于自己本身在正常文档流中的位置进行定位 相对它原来的位置,在走100px.原来在标准流中的位置继续占有. absolute: 生成绝对定位,相对于最近一级定位不为s ...
- 薅 AWS 羊毛的船新方式,以 ChatBot 为例
还在担心一年免费服务器到期后该怎么办?(Solo社区 投稿) 网上绝大多数薅 AWS 羊毛的教程都是在教大家如何申请创建一年免费的 VPS,太 OUT 了!就问一个问题,一年到期了那咋办? 其实,除了 ...