在 Spring Boot 中优雅地停止正在运行的函数并启动新函数

在 spring boot 中优雅地停止正在运行的函数并启动新函数

摘要:本文旨在提供一种在 Spring Boot 应用中优雅地停止长时间运行的任务,并允许启动新任务的方法。通过使用线程管理和唯一标识符,我们可以安全地中断正在执行的任务,避免资源浪费和潜在的并发问题。本文将提供详细的代码示例和解释,帮助开发者理解和实现这一功能。

在 Spring Boot 应用中,有时我们需要执行一些长时间运行的任务,例如日志记录、数据处理等。这些任务通常在一个独立的线程中运行,以避免阻塞主线程。然而,在某些情况下,我们可能需要停止这些正在运行的任务,并启动新的任务。以下提供一种使用线程和唯一 ID 来管理和停止后台任务的有效方法。

使用线程和唯一 ID 管理任务

核心思想是创建一个后台线程,并将其引用(以及可选的唯一ID)保存在一个数据结构中。当需要停止任务时,可以通过ID获取线程引用,并中断它。

1. 创建线程安全的线程存储

首先,我们需要一个线程安全的数据结构来存储线程的引用。ConcurrentHashMap 是一个不错的选择,因为它允许多个线程同时访问和修改它。

import java.util.Map;import java.util.UUID;import java.util.concurrent.ConcurrentHashMap;private static volatile Map threadlookup = new ConcurrentHashMap();

threadlookup 是一个静态的 ConcurrentHashMap,用于存储线程的唯一标识符 (UUID) 和线程对象之间的映射关系。volatile 关键字确保了多线程环境下的可见性。

2. 启动任务并存储线程信息

当收到启动任务的请求时,创建一个新的线程,启动它,并将线程的 UUID 和线程对象存储到 threadlookup 中。

import org.springframework.web.bind.annotation.GetMapping;import org.springframework.web.bind.annotation.RestController;@RestControllerpublic class TaskController {    private static volatile Map threadlookup = new ConcurrentHashMap();    @GetMapping("/start")    public String startTask() {        UUID uuid = UUID.randomUUID();        Thread thread = new Thread(() -> {            // 这里是长时间运行的任务            while (!Thread.currentThread().isInterrupted()) {                try {                    System.out.println("Task running... UUID: " + uuid);                    Thread.sleep(1000); // 模拟任务执行                } catch (InterruptedException e) {                    Thread.currentThread().interrupt(); // 重新设置中断状态                    System.out.println("Task interrupted. UUID: " + uuid);                }            }        });        thread.start();        threadlookup.put(uuid.toString(), thread);        return uuid.toString(); // 返回 UUID    }}

在上面的代码中:

UUID.randomUUID() 生成一个唯一的 UUID,用于标识该线程。Thread thread = new Thread(() -> { … }); 创建一个新的线程,并在 run() 方法中定义任务的逻辑。thread.start(); 启动线程。threadlookup.put(uuid.toString(), thread); 将 UUID 和线程对象存储到 threadlookup 中。最后,返回 UUID,以便客户端在停止任务时使用。在任务的 run() 方法中,使用 Thread.currentThread().isInterrupted() 检查线程是否被中断。如果被中断,则抛出 InterruptedException 异常,并在 catch 块中重新设置中断状态 (Thread.currentThread().interrupt();),确保线程能够正确地退出。

3. 停止任务

当收到停止任务的请求时,根据 UUID 从 threadlookup 中获取线程对象,并中断它。

import org.springframework.http.HttpStatus;import org.springframework.http.ResponseEntity;import org.springframework.web.bind.annotation.GetMapping;import org.springframework.web.bind.annotation.PathVariable;import org.springframework.web.bind.annotation.RestController;@RestControllerpublic class TaskController {    private static volatile Map threadlookup = new ConcurrentHashMap();    // startTask() 方法省略    @GetMapping("/stop/{uuid}")    public ResponseEntity stopTask(@PathVariable String uuid) {        Thread thread = threadlookup.get(uuid);        if (thread == null) {            return new ResponseEntity("Thread not found with UUID: " + uuid, HttpStatus.NOT_FOUND);        } else {            thread.interrupt();            threadlookup.remove(uuid); // 从 map 中移除            return new ResponseEntity("Thread interrupted with UUID: " + uuid, HttpStatus.OK);        }    }}

在上面的代码中:

@PathVariable String uuid 从请求路径中获取 UUID。threadlookup.get(uuid) 根据 UUID 从 threadlookup 中获取线程对象。如果线程对象不存在,则返回 404 错误。thread.interrupt(); 中断线程。 interrupt() 方法会设置线程的中断状态,并抛出 InterruptedException 异常(如果线程正在 sleep() 或 wait() 中)。threadlookup.remove(uuid); 从 threadlookup 中移除线程对象,释放资源。

完整示例代码

import org.springframework.boot.SpringApplication;import org.springframework.boot.autoconfigure.SpringBootApplication;import org.springframework.http.HttpStatus;import org.springframework.http.ResponseEntity;import org.springframework.web.bind.annotation.GetMapping;import org.springframework.web.bind.annotation.PathVariable;import org.springframework.web.bind.annotation.RestController;import java.util.Map;import java.util.UUID;import java.util.concurrent.ConcurrentHashMap;@SpringBootApplicationpublic class SpringBootTaskManagementApplication {    public static void main(String[] args) {        SpringApplication.run(SpringBootTaskManagementApplication.class, args);    }    @RestController    public static class TaskController {        private static volatile Map threadlookup = new ConcurrentHashMap();        @GetMapping("/start")        public String startTask() {            UUID uuid = UUID.randomUUID();            Thread thread = new Thread(() -> {                // 这里是长时间运行的任务                while (!Thread.currentThread().isInterrupted()) {                    try {                        System.out.println("Task running... UUID: " + uuid);                        Thread.sleep(1000); // 模拟任务执行                    } catch (InterruptedException e) {                        Thread.currentThread().interrupt(); // 重新设置中断状态                        System.out.println("Task interrupted. UUID: " + uuid);                    }                }            });            thread.start();            threadlookup.put(uuid.toString(), thread);            return uuid.toString(); // 返回 UUID        }        @GetMapping("/stop/{uuid}")        public ResponseEntity stopTask(@PathVariable String uuid) {            Thread thread = threadlookup.get(uuid);            if (thread == null) {                return new ResponseEntity("Thread not found with UUID: " + uuid, HttpStatus.NOT_FOUND);            } else {                thread.interrupt();                threadlookup.remove(uuid); // 从 map 中移除                return new ResponseEntity("Thread interrupted with UUID: " + uuid, HttpStatus.OK);            }        }    }}

注意事项

异常处理: 确保在任务的 run() 方法中正确处理 InterruptedException 异常,并在捕获异常后重新设置中断状态。资源释放: 在线程中断后,及时释放占用的资源,例如关闭文件流、数据库连接等。线程安全: 使用线程安全的数据结构来存储线程的引用,避免并发问题。单一线程场景: 如果只需要一个线程运行,则可以使用单个变量来存储线程引用,而无需使用 Map。

总结

通过使用线程管理和唯一标识符,我们可以优雅地停止 Spring Boot 应用中长时间运行的任务,并允许启动新的任务。这种方法可以避免资源浪费和潜在的并发问题,提高应用的稳定性和可靠性。 记住在实际应用中,需要根据具体的业务逻辑进行适当的调整和优化。

以上就是在 Spring Boot 中优雅地停止正在运行的函数并启动新函数的详细内容,更多请关注创想鸟其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。
如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至 chuangxiangniao@163.com 举报,一经查实,本站将立刻删除。
发布者:程序猿,转转请注明出处:https://www.chuangxiangniao.com/p/1581000.html

(0)
打赏 微信扫一扫 微信扫一扫 支付宝扫一扫 支付宝扫一扫
上一篇 2025年12月22日 21:57:04
下一篇 2025年12月22日 21:57:17

相关推荐

  • Spring Boot中长时任务的启动与安全中断管理

    本教程详细阐述了在Spring Boot应用中如何优雅地启动和停止长时间运行的后台任务,以避免阻塞主线程。核心策略是利用Java的线程机制,将耗时操作封装到独立的后台线程中执行,并通过一个全局映射管理这些线程。当需要停止任务时,向目标线程发送中断信号,并在任务内部适时响应此信号,从而实现任务的安全终…

    2025年12月22日
    000
  • SCSS嵌套失效问题排查与解决方案:深入理解SCSS类选择器

    第一段引用上面的摘要: 本文旨在帮助开发者理解SCSS嵌套规则,并解决在使用两层嵌套时样式无法生效的问题。通过分析常见错误原因,结合实际代码示例,详细讲解如何正确使用SCSS嵌套,避免选择器错误,从而编写出高效、可维护的CSS代码。本文重点在于解释SCSS嵌套生成的CSS选择器与HTML结构之间的关…

    2025年12月22日
    000
  • HTMLHTML5DragDropAPI拖拽功能的格式实现和事件处理

    要实现HTML5拖拽功能,需设置draggable=”true”并监听dragstart、dragover和drop等事件,在dragstart中通过dataTransfer.setData()传递数据,dragover中调用preventDefault()允许放置,drop…

    2025年12月22日
    000
  • 使用 CSS :last-of-type 高效管理相邻元素样式

    本教程旨在解决为一组相同类型元素中,紧邻不同类型元素的前一个元素添加特定样式的问题。针对初学者常倾向于使用 JavaScript 的情况,我们重点介绍并演示了如何利用 CSS 的 :last-of-type 伪类选择器,结合可选的容器元素,实现声明式、高性能的样式控制,从而避免复杂的 JavaScr…

    2025年12月22日 好文分享
    000
  • 解决下拉菜单外的区域禁用页面滚动的问题

    问题概述 正如摘要所说,当使用固定定位的遮罩层(page_blocker)来检测下拉菜单外部点击时,可能会导致在遮罩层区域滚动页面失效。下面我们详细分析这个问题,并给出解决方案。 问题分析 问题的原因在于 page_blocker 元素使用了 position: fixed;。当一个元素被设置为固定…

    2025年12月22日
    000
  • 文章主体内容应该放在哪里?MAIN和ARTICLE标签的核心作用。

    答案是ARTICLE标签用于包裹独立内容,MAIN标签定义页面核心区域,通常将ARTICLE置于MAIN内以提升语义化、可访问性与SEO。 文章主体内容应当放在 ARTICLE 标签内,这是HTML5中专门用于包裹独立、自包含内容的语义化标签。例如新闻报道、博客文章、评论等可独立传播的内容都适合用 …

    2025年12月22日
    000
  • 利用CSS :last-of-type 高效管理相邻元素样式

    本文探讨了如何为一组特定元素中的最后一个元素,当其后跟随不同类型元素时,添加特定样式类。通过优化HTML结构,将相关元素进行分组,并结合CSS的:last-of-type伪类选择器,能够以简洁、高效且易于维护的方式实现这一需求,避免了复杂的JavaScript逻辑。 在前端开发中,我们经常会遇到这样…

    2025年12月22日
    000
  • 在 Angular 项目中集成 Bootstrap Icons 的完整教程

    本教程详细介绍了如何在 Angular 项目中正确集成 Bootstrap Icons。通过 npm 安装依赖后,关键步骤是在 angular.json 配置文件中引入 Bootstrap Icons 的 CSS 样式路径,确保构建系统能够识别并加载图标字体,从而在 Angular 应用中顺利使用各…

    2025年12月22日
    000
  • 动态导航元素显示/隐藏的JavaScript最佳实践

    本文探讨了使用JavaScript实现动态导航元素显示与隐藏的多种方法。从最初仅隐藏相邻元素导致内容堆叠的问题,逐步优化到通过显式隐藏所有非目标元素,再到利用事件委托和自定义数据属性实现高效、可扩展且易于维护的解决方案,旨在提供构建响应式UI的专业指导。 在现代web开发中,实现交互式导航和动态内容…

    2025年12月22日
    000
  • 在 Angular 应用中动态刷新 Prism.js 语法高亮

    本文旨在解决 Angular 应用中,从数据库加载动态代码内容后,Prism.js 语法高亮无法自动更新的问题。通过利用 Prism.highlightElement() 方法,结合 Angular 的数据绑定和生命周期钩子,实现对特定代码块的精准高效刷新,确保动态加载的代码始终以正确的语法高亮形式…

    2025年12月22日 好文分享
    000
  • JavaScript实现动态导航栏元素显示与隐藏的优化教程

    本文详细探讨了如何通过JavaScript高效管理导航栏元素的显示与隐藏,避免内容堆叠问题。从基础的逐个控制到利用DOM缓存、集中化逻辑,最终引出事件委托与数据属性的现代化解决方案,旨在提供一个可扩展、高性能且易于维护的前端交互模式。 在前端开发中,动态显示和隐藏页面元素是常见的交互需求,尤其是在构…

    2025年12月22日
    000
  • Spring Boot 中终止并重启后台任务的实现方法

    在 Spring Boot 中终止并重启后台任务的实现方法 摘要:本文介绍了如何在 Spring Boot 应用中优雅地终止正在运行的后台任务,并启动新的任务。通过维护一个线程池和唯一的任务ID,可以实现对特定任务的精确控制,并避免资源浪费和潜在的并发问题。本文提供了示例代码,展示了如何使用 UUI…

    2025年12月22日
    000
  • Python使用BeautifulSoup从嵌套HTML中提取带继承样式的文本

    本教程将指导您如何使用Python和BeautifulSoup库,从包含嵌套标签的HTML字符串中,递归地提取所有文本片段及其计算后的CSS样式属性。文章通过一个实用的递归函数,详细讲解了如何处理样式继承,最终生成一个包含文本和对应样式的字典列表,适用于需要精细化文本样式分析的场景。 在处理复杂的h…

    2025年12月22日
    000
  • React与原生JavaScript中动态创建元素事件绑定失效问题解析与最佳实践

    当在React或原生JavaScript中动态插入HTML字符串时,传统的onClick事件绑定可能失效,导致ReferenceError。本文将深入解析此问题,并提供在React中利用JSX和合成事件、在原生JS中利用addEventListener的正确解决方案,确保动态元素的事件功能正常运作。…

    2025年12月22日
    000
  • 在 Angular 中动态更新 Prism.js 语法高亮代码块的实践指南

    本文将详细介绍如何在 Angular 应用中,当从数据库加载新代码字符串时,有效地刷新和更新 Prism.js 语法高亮的 textarea 和 元素。核心方法包括通过 FormControl 更新 textarea 内容,并利用 Prism.highlightElement() 精确地重新高亮特定…

    2025年12月22日
    000
  • 使用 SCSS 实现两层嵌套样式不生效问题解析与解决方案

    第一段引用上面的摘要:本文旨在解决 SCSS 两层嵌套样式不生效的问题。通过分析 SCSS 嵌套规则,明确选择器的含义,并提供修改 HTML 结构或 SCSS 代码的两种解决方案,帮助开发者正确使用 SCSS 嵌套功能,提高样式编写效率。本文重点强调了类选择器的精确匹配原则,并提供可行的代码示例。 …

    2025年12月22日
    000
  • 解决 Canvas API 坐标错位问题:精准定位绘图元素

    本文旨在解决在使用 Canvas API 进行绘图时遇到的坐标错位问题。通过分析常见的错误原因,并结合实际代码示例,详细讲解如何正确获取 Canvas 元素的偏移量,以及如何处理 Canvas 元素自身尺寸与绘图上下文尺寸不一致的情况,确保绘图操作能够精准地落在鼠标点击的位置。 在使用 HTML5 …

    2025年12月22日
    000
  • 如何在React或原生JS中正确处理动态创建元素的点击事件

    本文深入探讨了在React或原生JavaScript中动态创建按钮时,onclick事件无法正确触发ReferenceError的问题。核心在于理解React的虚拟DOM与原生HTML事件绑定机制的区别。教程将分别提供React组件内使用JSX绑定事件的最佳实践,以及在原生JS中通过正确onclic…

    2025年12月22日
    000
  • SVG 动画教程:实现线条和圆形振动效果

    本教程旨在指导读者如何使用 SVG 和 SMIL 动画,实现线条和圆形产生振动或摆动效果。通过将线条转换为贝塞尔曲线,并结合 animate 元素,可以模拟线条的弯曲和振动。同时,通过动画控制圆形的位置,使其与线条的末端保持同步运动,从而实现整体的动态效果。此外,还介绍了如何在圆形内部嵌入图像,并使…

    2025年12月22日
    000
  • 解决单页应用中Chrome浏览器回退后document.title不更新的挑战

    本文探讨了单页应用在Chrome浏览器中,通过document.title设置页面标题后,在用户执行浏览器回退操作时,标签页标题未能正确更新的特定问题。文章提供了一种两步解决方案:首先,在设置目标标题前先将其设置为空字符串或临时值;其次,在浏览器历史回退操作完成后重新应用标题,以确保标签页显示与do…

    2025年12月22日
    000

发表回复

登录后才能评论
关注微信