2021 面试还不知道如何优雅关闭Java线程?(上)

简介: 线程池提供了两个方法:shutdown()和shutdownNow():shutdown()是种保守的关闭线程池方法。线程池执行shutdown()后,就会拒绝接收新任务,但会等待线程池中正在执行的任务和已进入阻塞队列的任务,都执行完后才最终关闭线程池。shutdownNow()相对激进,线程池执行shutdownNow()后,会拒绝接收新任务,同时中断线程池中正在执行的任务,已进入阻塞队列的任务也会被剥夺了执行的机会,不过这些被剥夺执行机会的任务会作为shutdownNow()方法的返回值返回。因为shutdownNow()方法会中断正在执行的线程,所以提交到线程池的任务。

为什么需要线程的取消机制?

开启一个线程很容易。绝大多数时间,我们都会让它们自己运行直到结束。

但有时,我们希望提前结束线程。

哪些情况会需要提前结束呢?

用户请求取消

用户点击前端的“取消”按钮或接口调用发出取消请求(例如JMX)

有时间限制的操作

比如,某应用需要在有限时间内搜索问题空间,并在这个时间内选择最佳的解决方案。当计时器超时时,需要取消所有正在搜索的任务

应用程序事件

例如,应用程序对某个问题空间进行分解并搜索,从而使不同的任务可以搜索问题空间中的不同区域。当其中一一个任务找到了解决方案时,所有其他仍在搜索的任务都将被取消

错误

网页爬虫程序搜索相关的页面,并将页面或摘要数据保存到硬盘。当一个爬虫任务

发生错误时(例如,磁盘空间已满),那么所有搜索任务都会取消,此时可能会记录它们的当前状态,以便稍后重新启动

关闭

当一个程序或服务关闭时,必须对正在处理和等待处理的工作执行某种操作。在平

缓的关闭过程中,当前正在执行的任务将继续执行直到完成,而在立即关闭过程中,当前的任务则可能取消。

在Java中没有安全的抢占式方法停止线程,只有一些协作式的机制,使请求取消的任务和代码都遵循一种既定协议。


其中一种协作机制能设置某个“已请求取消(Cancellation Requested)” 标志,而任务将定

期地查看该标志。如果设置了这个标志,那么任务将提前结束。


要使任务和线程能安全、快速、可靠地停止下来,并非易事。Java 没有提供任何机制来安全地终止线程。


曾经的 Thread.stop 和 suspend 问题很大,禁止使用!


但Java提供了中断(Interruption), 这是一种协作机制,能够使一个线程终止另一个线程的当前工作。


这种协作应用场景是很多的,我们其实很少会希望某个任务、线程或服务立即停止,因为这种立即停止会使共享的数据结构处于不一致状态。

在编写任务和服务时可以使用这样的协作:当需要停止时,首先会清除当前正在执行的工作,然后再结束。这提供了更好的灵活性,因为任务本身的代码比发出取消请求的代码更清楚如何执行清理工作。


生命周期结束(End-of-Lifecycle) 的问题会使任务、服务以及程序的设计和实现等过程变

得复杂,而这个在程序设计中非常重要的要素却经常被忽略。

行为良好的软件与勉强运行的软件之间的最主要区别就是,行为良好的软件能很完善地处理失败、关闭和取消等过程。


有哪些任务取消的方案呢?


其实一种就是“标记位”。例如使用volatile类型的域保存取消状态标识。


代码如下:

3.png

一个可取消的任务必须有取消策略(CancellationPolicy),在这个策略中将详细定义:


其他代码如何(How)请求取消该任务

任务在何时(When)检查是否已经请求了取消

在响应取消请求时应该执行哪些(What) 操作

考虑现实世界中停止支付(Stop-Payment) 支票的示例。银行通常都会规定如何提交一个

停止支付的请求,在处理这些请求时需要做出哪些响应性保证,以及当支付中断后需要遵守哪些流程(例如通知该事务中涉及的其他银行,以及对付款人的账户进行费用评估)。这些流程和保证放在一起就构成了支票支付的取消策略。


PrimeGenerator使用了一种简单的取消策略:客户代码通过调用cancel来请求取消,

PrimeGenerator在每次搜索素数前首先检查是否存在取消请求,若存在则退出。



目录
相关文章
|
3天前
|
Java 开发者 C++
Java多线程同步大揭秘:synchronized与Lock的终极对决!
Java多线程同步大揭秘:synchronized与Lock的终极对决!
28 5
|
5天前
|
Java 调度 数据库
Java并发编程:深入理解线程池
在Java并发编程的海洋中,线程池是一艘强大的船,它不仅提高了性能,还简化了代码结构。本文将带你潜入线程池的深海,探索其核心组件、工作原理及如何高效利用线程池来优化你的并发应用。
|
5天前
|
监控 Java 测试技术
Java并发编程最佳实践:设计高性能的多线程系统
Java并发编程最佳实践:设计高性能的多线程系统
18 1
|
5天前
|
存储 监控 Java
Java多线程优化:提高线程池性能的技巧与实践
Java多线程优化:提高线程池性能的技巧与实践
22 1
|
5天前
|
Oracle Java API
虚拟线程:Java的新利器?
作者通过本文梳理了自己对虚拟线程的理解,顺便捋一捋Java线程的过去、现在和未来。ps:写这篇文章的时候,Java 19刚刚发布,而现在Java 21已经正式GA了,不过虚拟线程的API和底层实现并没有什么变化。
|
6天前
|
Java 程序员 调度
【JAVA 并发秘籍】进程、线程、协程:揭秘并发编程的终极武器!
【8月更文挑战第25天】本文以问答形式深入探讨了并发编程中的核心概念——进程、线程与协程,并详细介绍了它们在Java中的应用。文章不仅解释了每个概念的基本原理及其差异,还提供了实用的示例代码,帮助读者理解如何在Java环境中实现这些并发机制。无论你是希望提高编程技能的专业开发者,还是准备技术面试的求职者,都能从本文获得有价值的见解。
24 1
|
5天前
|
设计模式 安全 Java
Java并发编程实战:使用synchronized关键字实现线程安全
Java并发编程实战:使用synchronized关键字实现线程安全
17 0
|
3天前
|
安全 Java 开发者
Java多线程同步:synchronized与Lock的“爱恨情仇”!
Java多线程同步:synchronized与Lock的“爱恨情仇”!
35 5
|
3天前
|
Java 程序员
从0到1,手把手教你玩转Java多线程同步!
从0到1,手把手教你玩转Java多线程同步!
13 3
|
2天前
|
Java
在Java多线程领域,精通Lock接口是成为高手的关键。
在Java多线程领域,精通Lock接口是成为高手的关键。相较于传统的`synchronized`,Lock接口自Java 5.0起提供了更灵活的线程同步机制,包括可中断等待、超时等待及公平锁选择等高级功能。本文通过实战演练介绍Lock接口的核心实现——ReentrantLock,并演示如何使用Condition进行精确线程控制,帮助你掌握这一武林秘籍,成为Java多线程领域的盟主。示例代码展示了ReentrantLock的基本用法及Condition在生产者-消费者模式中的应用,助你提升程序效率和稳定性。
10 2
下一篇
云函数