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在每次搜索素数前首先检查是否存在取消请求,若存在则退出。



目录
相关文章
|
1天前
|
缓存 Java
Java并发编程:深入理解线程池
【5月更文挑战第26天】本文将深入探讨Java并发编程中的一个重要概念——线程池。我们将了解线程池的基本概念,它的工作原理,以及如何在Java中使用线程池。此外,我们还将讨论线程池的优点和缺点,以及在实际应用中如何选择合适的线程池策略。
|
1天前
|
Java 开发者
Java中的多线程编程:基础知识与实践
【5月更文挑战第26天】 在现代计算机科学中,多线程编程是一个复杂且强大的概念。Java作为一种广泛使用的编程语言,其对多线程的支持使得开发者能够创建出更为复杂、高效和响应迅速的应用程序。本文将深入探讨Java中的多线程编程,从基础概念到实际编程技巧,为读者提供全面的理解和实践指导。
|
1天前
|
Java 调度 开发者
Java并发编程:深入理解线程池
【5月更文挑战第26天】在Java中,线程池是处理并发任务的高效工具。本文将深入探讨Java线程池的原理、使用及性能优化,帮助开发者更好地理解和应用线程池技术。
|
1天前
|
安全 Java 开发者
探索Java中的多线程编程与并发控制
多线程编程是Java编程中不可或缺的一部分,它允许程序同时执行多个任务,从而显著提高程序的整体性能。然而,多线程编程也带来了诸如数据不一致、死锁等并发问题。本文将深入探讨Java中的多线程编程技术,包括线程的创建、同步与通信,并介绍几种常用的并发控制策略,旨在帮助读者更好地理解并发编程的复杂性和挑战,并学会如何编写高效、安全的并发程序。
|
1天前
|
安全 算法 Java
深入理解Java并发编程:线程安全与性能优化
【5月更文挑战第26天】在Java开发中,高效地处理并发编程问题对于提升应用的性能和稳定性至关重要。本文将深入探讨Java中的线程安全性问题,分析并发编程的挑战,并介绍一系列实用的解决方案和性能优化技巧。我们将通过具体示例,包括锁机制、线程池以及原子类等高级特性,来指导开发者如何在实践中实现高效的并发控制。
|
2天前
|
Java 开发者
Java中的多线程编程:理解、实现与性能优化
【5月更文挑战第25天】 在Java中,多线程编程是实现并发执行任务的关键手段。本文将深入探讨Java多线程的核心概念,包括线程的创建、生命周期、同步机制以及高级特性。我们将通过实例演示如何有效地创建和管理线程,同时着重分析多线程环境下的性能调优策略和常见问题解决方法。文章旨在为读者提供一个全面的视角,帮助其掌握Java多线程编程的技巧,并在实际开发中避免潜在的并发问题,提升程序的性能和稳定性。
|
2天前
|
安全 Java 测试技术
Java并发编程:理解线程同步和锁
【5月更文挑战第25天】本文深入探讨了Java并发编程的核心概念,即线程同步和锁。通过详细解释这两种机制,我们能够理解它们如何帮助解决多线程环境中的竞态条件问题。此外,文章还提供了一些示例代码,以展示如何在Java中使用这些技术。
|
2天前
|
安全 Java
Java中的多线程编程:从理论到实践
【5月更文挑战第25天】本文将深入探讨Java中的多线程编程,从基础理论到实际应用,为读者提供全面的理解和实践指导。我们将首先介绍多线程的基本概念和重要性,然后详细解析Java中实现多线程的两种主要方式:继承Thread类和实现Runnable接口。接着,我们将探讨如何在多线程环境中处理共享资源,以及如何利用Java的并发工具来管理线程。最后,我们将通过一个实际的例子来演示如何在实际开发中使用多线程。
|
2天前
|
Java 调度
Java并发编程:深入理解线程池
【5月更文挑战第25天】在Java中,线程池是一种管理线程的高效方式,它可以提高系统性能并降低资源消耗。本文将深入探讨Java线程池的原理、使用及优化方法,帮助读者更好地理解和应用线程池技术。
|
2天前
|
Java 调度
Java中线程池的技术
Java中线程池的技术
16 3