SpringBoot中ThreadPoolTaskExecutor的使用

2023-10-26

1 配置自己的线程池

@Configuration
@EnableAsync
public class ThreadPoolConfig
{
	// 核心线程池大小
    private int corePoolSize = 50;

    // 最大可创建的线程数
    private int maxPoolSize = 200;

    // 队列最大长度
    private int queueCapacity = 1000;

    // 线程池维护线程所允许的空闲时间
    private int keepAliveSeconds = 300;

    @Bean(name = "threadPoolTaskExecutor")
    public ThreadPoolTaskExecutor threadPoolTaskExecutor()
    {
        ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
        // 设置最大线程池数量
        executor.setMaxPoolSize(maxPoolSize);
        // 核心线程数
        executor.setCorePoolSize(corePoolSize);
        // 线程池维护线程所允许的空闲时间
        executor.setKeepAliveSeconds(keepAliveSeconds);
        // 队列最大长度
        executor.setQueueCapacity(queueCapacity);
        // 线程前缀名称
        executor.setThreadNamePrefix("ThreadPoolTaskExecutor======>");
        // 设置线程池关闭时,等待所有任务都完成后,再继续销毁其他的bean
        // 这样这些异步任务的销毁就会优先于 数据库连接池对象的销毁
        // 也就是说当其他bean都完成任务销毁后,数据库连接池对象才会销毁
        executor.setWaitForTasksToCompleteOnShutdown(true);
        // 任务的等待时间,如果超过这个时间还没有销毁,就强制销毁,以确保应用最后能够被关闭,而不是阻塞住
        executor.setAwaitTerminationSeconds(5);
        // 线程池对拒绝任务(无线程可用)的处理策略
        // 线程不够用时由调用的线程处理该任务
        executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
        executor.initialize();
        return executor;
    }
}
  • 一定要自己配置,不要使用SpringBoot自带的,Spring默认的线程池没有上限,并发多时,可能会OOM。并且不是线程重用,每次都会新建一个线程。ThreadPoolTaskExecutor 会重用线程,节省创建线程的资源。因此,在使用ThreadLocal时,也要及时清理缓存,否则,会获取到上一个任务的参数。

2 使用

2.1 在Service层使用

  • Controller层直接返回结果,具体的业务,由其他线程执行。
	@Override
    @Async("threadPoolTaskExecutor")
    public void concatAB() {
    	......
    }

2.2 多线程中使用事务的写法

  • 事务放到其他方法@Transactional,然后调用。
	@Override
    @Async("threadPoolTaskExecutor")
    public void concatAB() {
    	addChange()
    	......
    }

 @Transactional(rollbackFor = Exception.class,isolation = Isolation.SERIALIZABLE)
    public void addChange(){
		.......
	}

2.3 方法内多线程

2.3.1 错误写法

  • 错误写法
	@Override
    public void concatAB() {
    	addChange()
    	......
    }

    @Async("threadPoolTaskExecutor")
    public void addChange(){
		.......
	}
  • @Async注解可以让Spring使用线程池来异步执行带有该注解的方法,当一个带有@Async注解的方法被调用时,Spring会为其创建代理对象。通过这个代理对象来调用该方法,Spring就可以控制方法的异步执行了。

  • 如果不是通过代理对象来调用@Async方法,方法的执行将与当前线程同步运行,不会实现异步处理的效果。所以在同一个类中调用加了@Async注解的方法,需要使用代理对象来调用以确保异步生效。

2.3.2 正确写法(一)

  • 第一种,直接使用线程池执行异步方法。
	@Autowired
    private ThreadPoolTaskExecutor threadPoolTaskExecutor;
	
	@Override
    public void concatAB() {
    	threadPoolTaskExecutor.execute(() -> {
            ....
        });
    }

2.3.2 正确写法(二)

  • 使用代理对象
@Override
    public void concatAB() {
        // 获取代理对象
    	IDemoService demoService = (IDemoService) AopContext.currentProxy();
    	// 使用代理对象调用
    	demoService.addChange()
    	......
    }

    @Async("threadPoolTaskExecutor")
    public void addChange(){
		.......
	}

2.3.3 正确写法(三)

  • DemoService中通过@Autowired将自身注入到selfProxy属性中,并使用selfProxy对象在syncMethod方法中调用了asyncMethod方法,从而实现了对同一类中的带有@Async注解的方法的异步调用。
@Autowired
    private DemoService selfProxy;

    @Async
    public void asyncMethod() {
        // 异步执行的逻辑
    }

    public void syncMethod() {
        // 调用异步方法需要使用代理对象
        selfProxy.asyncMethod();
    }

3 线程池与并行流的选取

  • IO密集型的业务,选取线程池来完成。(例如,连接数据库之类的操作)
  • cpu密集型的业务,选取并行流allelesvs.parallelStream()的方式处理。(例如,计算类、过滤、求和、分组等)
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

SpringBoot中ThreadPoolTaskExecutor的使用 的相关文章

  • Java中如何做系统捷径跨平台集成?

    您可能知道 Mac OS X 中保存的快捷键是Cmd S在 Windows 上是Ctrl S 关闭应用程序的捷径是Cmd QWindows 是Alt F4 但问题是如何在 java 应用程序中执行这些操作 我是否需要找到我在应用程序中使用的
  • GCP 日志浏览器显示日志记录的错误严重级别

    我正在 GKE 中运行 java 应用程序并在日志资源管理器中监视日志 Java 应用程序正在将日志写入stdout据我了解 GKE 代理解析它并将其发送到日志资源管理器 我发现日志浏览器显示WARN and ERROR具有严重性的消息IN
  • 从大表中检索所有记录时如何避免 OOM(内存不足)错误?

    我的任务是将一个巨大的表转换为自定义 XML 文件 我将使用 Java 来完成这项工作 如果我只是发出 SELECT FROM customer 它可能会返回大量数据 最终导致 OOM 我想知道 有没有一种方法可以在记录可用后立即处理该记录
  • 有人在实际应用程序中使用短和字节原始类型吗?

    我自 2004 年以来一直使用 Java 进行编程 主要是企业和 Web 应用程序 但我从来没有用过short or byte 而不是只是为了了解这些类型如何工作的玩具程序 即使在一个for loop100次 我们通常会选择int 我不记得
  • java:查找数组中整数的频率

    我需要开发一个java要求用户输入一些内容的程序integers并找出最大和最小的数 以及这些数的平均值 然后 划分数组的集合分成若干子区间用户指定的 然后它生成一个边界点 每个边界点的长度为子区间宽度 问题是我需要创建一个频率 例如 间隔
  • C++ 中的模块路径到 Java JNI 调用

    当我用 C 创建 Java 8 JVM 时 我通常使用类似以下代码的内容来告诉 JVM 类路径 JavaVMOption options new JavaVMOption 1 JVM invocation options options 0
  • 原生状态栏

    有没有办法创建nativeSWT 中的状态栏与 Windows 应用程序中的状态栏类似 我见过使用标签模拟的状态栏 但我对真正的解决方案更感兴趣 org eclipse jface action StatusLineManager crea
  • 验证在子类上调用此方法时是否调用了重写的超类方法

    我将用这个例子来展示我的问题 我有一个带有方法的类foo 该类有一个重写此方法的子类 子类的方法调用超类的方法 我可以验证一下吗 我不想测试什么foo在超类中确实如此 我只需要验证它是否被调用 我知道重构可以有所帮助 优先考虑组合而不是继承
  • 在 CSS 中设置 TableView 样式 (JavaFX)

    如何在 TableView 中设置 THIS 点的样式 我的 CSS 代码如下所示 Empty Stylesheet file root fx background color 262626 table view fx background
  • 无法加载(查找)j2v8_android_x86 库

    我有一个使用的 gradle 项目j2v8 android 2 2 1 http mvnrepository com artifact com eclipsesource j2v8 j2v8 android 2 2 1库 为 V8 JS 引
  • 来自公共字符串的 Android RSA 加密

    我正在开发一个 Android 应用程序 我希望用户能够使用其他人的公钥加密消息 系统将生成公钥 私钥对 然后可以将消息秘密发送给其他用户 我正在创建一个加密类 它将处理消息的加密 解密 不幸的是我遇到了一些问题 在这种方法中 我想传递用户
  • equals 和 hashcode 的不同字段

    我同意这篇文章的声明在Java中重写equals和hashCode时应该考虑哪些问题 https stackoverflow com questions 27581 overriding equals and hashcode in jav
  • 用于将类文件转换为 Java 源代码的 Eclipse 插件 [关闭]

    Closed 这个问题不符合堆栈溢出指南 help closed questions 目前不接受答案 当源不可用时 是否有可用的 Eclipse 插件来反编译类文件 从 class 文件生成 java 源代码 类似于 Resharper 在
  • 使用 Android 将文本文件上传到 Google Drive

    编辑 我已将文本设置为字符串 如下所示 字符串文本 你好 我想将其转换为纯文本文件 然后上传到 Google 云端硬盘文件夹 我已经尝试过下面的代码 但它不完整 所以我无法说出出现了什么错误 我正在使用 Google Drive 快速启动
  • java.lang.String 无法转换为 org.json.simple.JSONObject simple-json

    我在尝试使用 google 的 simple json 解析简单的 json 时遇到奇怪的问题 这是我的代码 它不起作用 String s args 0 toString JSONObject json JSONObject new JSO
  • while 循环无法访问代码

    当我编译这段代码时 public static void main String args int x 0 while false System out println hello 它显示编译时错误无法访问代码 但是当我将这段代码修改为 p
  • 如何在 SnakeYaml 中解析 YAML 文件的一部分

    我是 YAML 新手 并且解析了一个 YAML 配置文件 如下所示 applications authentication service version 2 0 service url https myapp corp auth app
  • 不要使用android内置的org.json

    我写了一个使用的库org json http json org A 来自 json org 假设 Android 使用相同的 在 android 中也称为org json B 只是它遗漏了一些相对关键的功能 现在我想做的是设置我的 grad
  • 使用 Wildfly 在 Jersey 上使用 @PersistenceUnit 注入 EntityManagerFactory

    我尝试使用 PersistenceUnit 注入 EntityManagerFactory 但它始终为空 我认为我的 persistence xml 没问题 因为我可以使用以下代码获取 EntityManager EntityManager
  • TextView 用字母打乱了我的话

    我的要求 创建 传入气泡 其宽度按内容排列 最大宽度为 90 我有这个标记

随机推荐