Spring框架使用xml方式配置ThreadPoolTaskExecutor线程池,并且自定义线程工厂

一、自定义线程工厂

自定义线程工厂需要实现java.util.concurrent.ThreadFactory 接口,重写newThread 方法。

示例代码:

java 复制代码
package com.xiaobai.thread;

import org.apache.log4j.Logger;

import java.util.concurrent.ThreadFactory;
import java.util.concurrent.atomic.AtomicInteger;

/**
 * 业务自定义线程工程
 * @author wangt
 */
public class BusinessThreadFactory implements ThreadFactory {

    private final Logger logger = Logger.getLogger(BusinessThreadFactory.class);

    // 线程组
    private final ThreadGroup group;

    // 线程序号
    private final AtomicInteger threadNumber = new AtomicInteger(1);

    // 线程名称前缀
    private final String namePrefix;

    public BusinessThreadFactory(String namePrefix) {
        SecurityManager securityManager = System.getSecurityManager();
        group = securityManager != null ? securityManager.getThreadGroup()
                : Thread.currentThread().getThreadGroup();

        this.namePrefix = namePrefix + "-thread-";
    }

    @Override
    public Thread newThread(Runnable runnable) {

        Thread thread = new Thread(group, runnable, namePrefix + threadNumber.getAndIncrement(), 0);

        // 守护线程
        if (thread.isDaemon()) {
            // 将当前线程设置为守护线程,主线程结束时,当前线程也随之结束
            thread.setDaemon(true);
        }

        // 线程优先级
        if (thread.getPriority() != Thread.NORM_PRIORITY) {
            // 设置普通优先级
            thread.setPriority(Thread.NORM_PRIORITY);
        }

        // 处理未捕捉的异常
        thread.setUncaughtExceptionHandler(new Thread.UncaughtExceptionHandler() {
            @Override
            public void uncaughtException(Thread t, Throwable e) {
                logger.error("异常:", e);
            }
        });

        return thread;
    }
}

二、applicationContext.xml配置自定义线程工厂及线程池

(一)自定义线程工厂配置

xml 复制代码
<!--自定义线程工厂-->
<bean id="businessThread" class="com.xiaobai.thread.BusinessThreadFactory" >
    <!--自定义线程工厂构造函数,参数为线程名称前缀-->
    <constructor-arg value="business" name="namePrefix"/>
</bean>

(二)配置ThreadPoolTaskExecutor线程池

xml 复制代码
<!--Spring线程池-->
<bean id="taskExecutor" class="org.springframework.scheduling.concurrent.ThreadPoolTaskExecutor">
    <!--核心线程数量-->
    <property name="corePoolSize" value="20" />
    <!--最大线程数量-->
    <property name="maxPoolSize" value="50" />
    <!--队列容量-->
    <property name="queueCapacity" value="100" />
    <!--允许的空闲时间-->
    <property name="keepAliveSeconds" value="60" />
    <!--线程超过空闲时间限制,会直接退出知道线程数量为0-->
    <property name="allowCoreThreadTimeOut" value="true"/>
    <!--配置自定义线程工厂-->
    <property name="threadFactory" ref="businessThread"/>
</bean>

(三)ThreadPoolTaskExecutor线程池核心参数配置说明:

1、corePoolSize(核心线程数量):

配置核心线程数量需要考虑任务类型和任务负载等因素。
(1)CPU密集型任务 :核心线程数量可设置为 CPU 核心数 + 1
(2)IO密集型任务 :核心线程数量可设置为CPU 核心数 × 2 ,IO操作会导致线程阻塞,更多的线程可提高并发能力;

(3)如果任务负载较轻,可适当减小核心线程数量,如果任务负载较重,可适当增加核心线程数量;

2、maxPoolSize(最大线程数量):

配置最大线程数量需要考虑任务类型和任务负载等因素,最大线程数量应与队列容量匹配。
(1)CPU密集型任务 :最大线程数量可设置为 CPU 核心数 + 1CPU 核心数 ×2 之间;
(2)IO密集型任务 :最大线程数量可设置为CPU核心数×2核心数×4之间;

3、queueCapacity(队列容量):

(1)CPU密集型任务 :队列容量通常设置为100;
(2)IO密集型任务:队列容量通常设置为200到500;

4、keepAliveTime(空闲线程存活时间):

一个线程如果处于空闲状态,在指定时间内,这个线程会被销毁。

5、allowCoreThreadTimeOut(是否会回收核心线程):

如果核心线程空闲时间超过keepAliveTime,核心线程线程会被回收。

6、rejectedExecutionHandler(拒绝策略):

默认拒绝策略是ThreadPoolExecutor.AbortPolicy,任务数量超过最大线程数量,抛出RejectedExecutionException 异常。

(四)线程池使用

1、注入线程池
java 复制代码
@Autowired
private ThreadPoolTaskExecutor taskExecutor;
2、使用线程池创建线程
java 复制代码
Runnable runnable = new Runnable() {
    @Override
    public void run() {
        testService.threadTest(user);
    }
};
taskExecutor.execute(runnable);

参考
ThreadPoolExecutor拒绝策略
为何自定义线程工厂(ThreadFactory)

相关推荐
暮乘白帝过重山16 分钟前
Singleton和Prototype的作用域与饿汉式/懒汉式的初始化方式
spring·原型模式·prototype·饿汉式·singleton·懒汉式
腥臭腐朽的日子熠熠生辉1 小时前
解决maven失效问题(现象:maven中只有jdk的工具包,没有springboot的包)
java·spring boot·maven
ejinxian1 小时前
Spring AI Alibaba 快速开发生成式 Java AI 应用
java·人工智能·spring
杉之1 小时前
SpringBlade 数据库字段的自动填充
java·笔记·学习·spring·tomcat
圈圈编码1 小时前
Spring Task 定时任务
java·前端·spring
俏布斯1 小时前
算法日常记录
java·算法·leetcode
27669582921 小时前
美团民宿 mtgsig 小程序 mtgsig1.2 分析
java·python·小程序·美团·mtgsig·mtgsig1.2·美团民宿
爱的叹息2 小时前
Java 连接 Redis 的驱动(Jedis、Lettuce、Redisson、Spring Data Redis)分类及对比
java·redis·spring
程序猿chen2 小时前
《JVM考古现场(十五):熵火燎原——从量子递归到热寂晶壁的代码涅槃》
java·jvm·git·后端·java-ee·区块链·量子计算
松韬2 小时前
Spring + Redisson:从 0 到 1 搭建高可用分布式缓存系统
java·redis·分布式·spring·缓存