Spring Boot 优雅关机时异步线程安全优化

2024-06-17 14:12

本文主要是介绍Spring Boot 优雅关机时异步线程安全优化,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

个人名片
在这里插入图片描述
🎓作者简介:java领域优质创作者
🌐个人主页:码农阿豪
📞工作室:新空间代码工作室(提供各种软件服务)
💌个人邮箱:[2435024119@qq.com]
📱个人微信:15279484656
🌐个人导航网站:www.forff.top
💡座右铭:总有人要赢。为什么不能是我呢?

  • 专栏导航:

码农阿豪系列专栏导航
面试专栏:收集了java相关高频面试题,面试实战总结🍻🎉🖥️
Spring5系列专栏:整理了Spring5重要知识点与实战演练,有案例可直接使用🚀🔧💻
Redis专栏:Redis从零到一学习分享,经验总结,案例实战💐📝💡
全栈系列专栏:海纳百川有容乃大,可能你想要的东西里面都有🤸🌱🚀

Spring Boot 优雅关机时异步线程安全优化

在现代微服务架构中,应用程序的可用性和稳定性至关重要。Spring Boot 提供了优雅关机(Graceful Shutdown)功能,以确保在服务关闭时能完成正在处理的请求并释放资源。然而,在涉及异步任务时,确保线程安全性和任务的正确终止尤为重要。本文将探讨如何在Spring Boot应用中实现优雅关机,并对异步线程进行安全优化。

为什么需要优雅关机?

优雅关机的主要目的是在关闭应用程序时,允许已经接收到的请求处理完成,而不再接受新的请求。这可以避免请求中途被打断,从而提供更好的用户体验。此外,它还允许释放资源,例如数据库连接和文件句柄,避免资源泄漏。

配置Spring Boot的优雅关机

Spring Boot从2.3.0版本开始支持优雅关机。通过在application.properties中配置以下参数,可以启用这一功能:

server.shutdown=graceful
spring.lifecycle.timeout-per-shutdown-phase=30s

这将使Spring Boot在接收到关闭信号后,有一个30秒的时间窗口来完成正在处理的请求。

异步任务的优雅关机

在涉及异步任务时,确保它们在关机期间正确地完成或中止是一个挑战。以下是一些关键步骤来优化异步任务的线程安全和优雅关机。

1. 使用 ThreadPoolTaskExecutor

首先,配置一个自定义的 ThreadPoolTaskExecutor 来管理异步任务。它提供了更灵活的线程池管理,并支持优雅关机。

import org.springframework.context.annotation.Bean;
import org.springframework.context.annotation.Configuration;
import org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor;import java.util.concurrent.Executor;@Configuration
public class AsyncConfig {@Bean(name = "taskExecutor")public Executor taskExecutor() {ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();executor.setCorePoolSize(5);executor.setMaxPoolSize(10);executor.setQueueCapacity(500);executor.setThreadNamePrefix("Async-");executor.setWaitForTasksToCompleteOnShutdown(true);executor.setAwaitTerminationSeconds(30);executor.initialize();return executor;}
}

在这里,我们设置了 waitForTasksToCompleteOnShutdowntrue,并指定了 awaitTerminationSeconds 为 30秒,这意味着在应用关闭时,线程池将等待最多30秒来完成当前正在执行的任务。

2. 使用 @Async 注解

然后,在需要异步执行的方法上使用 @Async 注解,并指定使用自定义的任务执行器。

import org.springframework.scheduling.annotation.Async;
import org.springframework.stereotype.Service;@Service
public class MyAsyncService {@Async("taskExecutor")public void executeAsyncTask() {// 任务逻辑try {Thread.sleep(5000);  // 模拟长时间运行的任务} catch (InterruptedException e) {Thread.currentThread().interrupt();}System.out.println("异步任务执行完成");}
}

通过这样配置,异步任务将在自定义的线程池中执行,并在应用关闭时遵循优雅关机的设置。

3. 确保任务的线程安全

在执行异步任务时,需要确保线程安全。常见的做法是使用线程安全的数据结构(例如 ConcurrentHashMap),或使用适当的同步机制(例如 synchronized 块)。

import java.util.concurrent.ConcurrentHashMap;
import java.util.concurrent.ConcurrentMap;@Service
public class ThreadSafeService {private final ConcurrentMap<String, String> dataMap = new ConcurrentHashMap<>();@Async("taskExecutor")public void updateData(String key, String value) {dataMap.put(key, value);}public String getData(String key) {return dataMap.get(key);}
}
4. 处理任务中断

在优雅关机过程中,异步任务可能会收到中断信号,因此需要正确处理中断异常。

@Async("taskExecutor")
public void executeInterruptibleTask() {try {// 模拟长时间运行的任务for (int i = 0; i < 10; i++) {Thread.sleep(1000);  // 每秒执行一次检查System.out.println("任务进行中...");}} catch (InterruptedException e) {Thread.currentThread().interrupt();  // 恢复中断状态System.out.println("任务被中断");}
}

通过捕获 InterruptedException 并恢复线程的中断状态,可以确保任务在收到中断信号时能够正确处理。

结论

在Spring Boot中实现优雅关机不仅可以提高应用的稳定性,还可以确保在关闭期间异步任务能够安全地完成或中止。通过配置 ThreadPoolTaskExecutor 和使用 @Async 注解,可以有效管理异步任务的执行。同时,通过使用线程安全的数据结构和正确处理中断信号,可以进一步优化异步任务的安全性和可靠性。希望本文能够帮助你在实际项目中更好地实现优雅关机和异步线程安全优化。

这篇关于Spring Boot 优雅关机时异步线程安全优化的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1069667

相关文章

Java五子棋之坐标校正

上篇针对了Java项目中的解构思维,在这篇内容中我们不妨从整体项目中拆解拿出一个非常重要的五子棋逻辑实现:坐标校正,我们如何使漫无目的鼠标点击变得有序化和可控化呢? 目录 一、从鼠标监听到获取坐标 1.MouseListener和MouseAdapter 2.mousePressed方法 二、坐标校正的具体实现方法 1.关于fillOval方法 2.坐标获取 3.坐标转换 4.坐

Spring Cloud:构建分布式系统的利器

引言 在当今的云计算和微服务架构时代,构建高效、可靠的分布式系统成为软件开发的重要任务。Spring Cloud 提供了一套完整的解决方案,帮助开发者快速构建分布式系统中的一些常见模式(例如配置管理、服务发现、断路器等)。本文将探讨 Spring Cloud 的定义、核心组件、应用场景以及未来的发展趋势。 什么是 Spring Cloud Spring Cloud 是一个基于 Spring

Javascript高级程序设计(第四版)--学习记录之变量、内存

原始值与引用值 原始值:简单的数据即基础数据类型,按值访问。 引用值:由多个值构成的对象即复杂数据类型,按引用访问。 动态属性 对于引用值而言,可以随时添加、修改和删除其属性和方法。 let person = new Object();person.name = 'Jason';person.age = 42;console.log(person.name,person.age);//'J

java8的新特性之一(Java Lambda表达式)

1:Java8的新特性 Lambda 表达式: 允许以更简洁的方式表示匿名函数(或称为闭包)。可以将Lambda表达式作为参数传递给方法或赋值给函数式接口类型的变量。 Stream API: 提供了一种处理集合数据的流式处理方式,支持函数式编程风格。 允许以声明性方式处理数据集合(如List、Set等)。提供了一系列操作,如map、filter、reduce等,以支持复杂的查询和转

uniapp接入微信小程序原生代码配置方案(优化版)

uniapp项目需要把微信小程序原生语法的功能代码嵌套过来,无需把原生代码转换为uniapp,可以配置拷贝的方式集成过来 1、拷贝代码包到src目录 2、vue.config.js中配置原生代码包直接拷贝到编译目录中 3、pages.json中配置分包目录,原生入口组件的路径 4、manifest.json中配置分包,使用原生组件 5、需要把原生代码包里的页面修改成组件的方

Java面试八股之怎么通过Java程序判断JVM是32位还是64位

怎么通过Java程序判断JVM是32位还是64位 可以通过Java程序内部检查系统属性来判断当前运行的JVM是32位还是64位。以下是一个简单的方法: public class JvmBitCheck {public static void main(String[] args) {String arch = System.getProperty("os.arch");String dataM

详细分析Springmvc中的@ModelAttribute基本知识(附Demo)

目录 前言1. 注解用法1.1 方法参数1.2 方法1.3 类 2. 注解场景2.1 表单参数2.2 AJAX请求2.3 文件上传 3. 实战4. 总结 前言 将请求参数绑定到模型对象上,或者在请求处理之前添加模型属性 可以在方法参数、方法或者类上使用 一般适用这几种场景: 表单处理:通过 @ModelAttribute 将表单数据绑定到模型对象上预处理逻辑:在请求处理之前

eclipse运行springboot项目,找不到主类

解决办法尝试了很多种,下载sts压缩包行不通。最后解决办法如图: help--->Eclipse Marketplace--->Popular--->找到Spring Tools 3---->Installed。

JAVA读取MongoDB中的二进制图片并显示在页面上

1:Jsp页面: <td><img src="${ctx}/mongoImg/show"></td> 2:xml配置: <?xml version="1.0" encoding="UTF-8"?><beans xmlns="http://www.springframework.org/schema/beans"xmlns:xsi="http://www.w3.org/2001

Java面试题:通过实例说明内连接、左外连接和右外连接的区别

在 SQL 中,连接(JOIN)用于在多个表之间组合行。最常用的连接类型是内连接(INNER JOIN)、左外连接(LEFT OUTER JOIN)和右外连接(RIGHT OUTER JOIN)。它们的主要区别在于它们如何处理表之间的匹配和不匹配行。下面是每种连接的详细说明和示例。 表示例 假设有两个表:Customers 和 Orders。 Customers CustomerIDCus