线程池正确打开方式- 拌饭使用效果更佳
为什么要用线程池?什么是线程池?怎么使用?素质三连!!!
- 为什么要用线程池?
- 节约系统资源。避免反复创建销毁线程
- 提供异步操作。提高响应速度
- 方便管理线程
- 什么是线程池?
- 多线程使用的一种管理方式。池化技术能有效的利用和管理一些资源
- 怎么使用?
- 当然是配合springboot使用。皮一下😂
- 整个过程梳理如下
- 让IOC容器加载和管理线程池 - 即配置完成之后,项目启动便加载配置
- 调用时使用注解 - 方便开发
线程池核心概念介绍
-
corePoolSize 正常持有的线程数。在使用线程时,如果池中的线程数小于该数,会创建新线程
-
maximumPoolSize 允许拥有的最大线程数。 池中核心线程core如果超出了,会放到阻塞队列中。队列如果也满了,会申请创建线程但数量不能超过该数值。如果超过了会执行拒绝策略
-
BlockingQueue<Runnable> workQueue 阻塞队列,用于存放等待资源的线程。
-
long keepAliveTime,TimeUnit unit 大于核心线程数的线程空闲之后的存活时间。活着浪费空气
实际配置如下
- 启动类上添加@EnableConfigurationProperties 注解。用于解析自定义配置属性。同时需要引入pom配置
-
@EnableConfigurationProperties public class MerchantApplication { public static void main(String[] args) { SpringApplication.run(MerchantApplication.class, args); } } <dependency> <groupId>org.springframework.boot</groupId> <artifactId>spring-boot-configuration-processor</artifactId> <optional>true</optional> </dependency>
-
-
配置启动加载时需要的类。1.yml配置属性。2.配置类启动加载
-
// yml配置 task: pool: corePoolSize: 5 queueCapacity: 20 maxPoolSize: 10 keepAliveSeconds: 3 // 属性映射类 @Data @ConfigurationProperties(prefix = "task.pool") @Component public class TaskThreadPoolProperty { private int corePoolSize; private int maxPoolSize; private int keepAliveSeconds; private int queueCapacity; }
// 启动加载@Configuration
@EnableAsync
public class ThreadPoolConfig {
@Resource
private TaskThreadPoolProperty config;
/**
* 默认使用 名为 taskExecutor 的线程池
*
* @return 初始化线程池
*/
@Bean("taskExecutor")
public Executor configTaskAsyncPool() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
//核心线程池大小
executor.setCorePoolSize(config.getCorePoolSize());
//最大线程数
executor.setMaxPoolSize(config.getMaxPoolSize());
//队列容量
executor.setQueueCapacity(config.getQueueCapacity());
//活跃时间
executor.setKeepAliveSeconds(config.getKeepAliveSeconds());
//线程名字前缀
executor.setThreadNamePrefix("Qc-");
// CallerRunsPolicy:不在新线程中执行任务,而是由调用者所在的线程来执行
executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
executor.setWaitForTasksToCompleteOnShutdown(true);
return executor;
}
}
-
- 调用 使用@Async注解调用
-
// 实现异步操作的类 @Component public class AsyncMessage { @Async public void sendMsgCL(String mobile, String content) { try { Thread.sleep(3000L); System.out.println("多线程异步执行" + " " + Thread.currentThread().getName()); } catch (InterruptedException e) { e.printStackTrace(); } } } // 测试异步的接口 @Resource private AsyncMessage message; @PostMapping("/data1") public void data() { message.sendMsgCL(null, null); System.out.println("haha"); }
-
- 注意点(如果不注意可能会让你多吃两碗)
- 配置yml自定义属性之后需要重启IDEA才能生效。
- 启动加载的bean命名为@Bean("taskExecutor") 。spring在调用线程池时默认的池子名称为taskExecutor
- 使用@Async注解可以指明使用的线程池
- 不能修饰静态static方法
- 异步类需要被IOC容器管理。即要使用注解类似@Component @Service 。。。等等
- 异步方法不能与被调用的异步方法在同一个类中
- @EnableAsync 记得加上。可以在启动类上也可以在配置类上。
是谁来自江河湖海,却囿于昼夜厨房与爱