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

心靈之曲
发布: 2025-10-04 13:42:01
原创
191人浏览过

在 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<String, Thread> threadlookup = new ConcurrentHashMap<>();
登录后复制

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

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

怪兽AI数字人
怪兽AI数字人

数字人短视频创作,数字人直播,实时驱动数字人

怪兽AI数字人 44
查看详情 怪兽AI数字人

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

import org.springframework.web.bind.annotation.GetMapping;
import org.springframework.web.bind.annotation.RestController;

@RestController
public class TaskController {

    private static volatile Map<String, Thread> 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;

@RestController
public class TaskController {

    private static volatile Map<String, Thread> threadlookup = new ConcurrentHashMap<>();

    // startTask() 方法省略

    @GetMapping("/stop/{uuid}")
    public ResponseEntity<String> 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;

@SpringBootApplication
public class SpringBootTaskManagementApplication {

    public static void main(String[] args) {
        SpringApplication.run(SpringBootTaskManagementApplication.class, args);
    }

    @RestController
    public static class TaskController {

        private static volatile Map<String, Thread> 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<String> 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 中优雅地停止正在运行的函数并启动新函数的详细内容,更多请关注php中文网其它相关文章!

最佳 Windows 性能的顶级免费优化软件
最佳 Windows 性能的顶级免费优化软件

每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。

下载
来源:php中文网
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
最新问题
开源免费商场系统广告
热门教程
更多>
最新下载
更多>
网站特效
网站源码
网站素材
前端模板
关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新 English
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送
PHP中文网APP
随时随地碎片化学习

Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号