一、前言
项目需要,加上本人没怎么用过线程池,异步线程这种,便想学习一下。
实现线程有两种,继承Thread和实现Runable接口。不了解的稍微去了解一下。
楼主一开始用的是实现Runable接口
这里楼主需要传递参数进去,所以重新定义了一个接口去实现
public interface MyRunable extends Runnable {
// 注意:如果想要传递参数给异步线程,则重新定义抽象方法去实现
void setParam(Object obj);
}
之后实现该接口
public class ThreadTask implements MyRunable {
private Object object;
@Override
public void setParam(Object obj) {
object = obj;
}
@Override
public void run() {
System.out.println("xiaoshishu >>> 传递进来的参数" + object.hashCode());
}
}
遇到的坑:因为需要增删改查,调用数据库那边,接口一直报空指针异常;
解决方法:直接定义一个工具类,通过类名直接去Spring上下文获取即可,但是这种方法我不太喜欢。也找不到好的解决方法,后面抛弃掉使用线程池。
2、配置线程池
整合线程池
@Configuration
@EnableAsync
public class ThreadPoolConfig implements AsyncConfigurer {
@Override
@Bean
public Executor getAsyncExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
// 设置核心线程数
executor.setCorePoolSize(5);
// 设置最大线程数
executor.setMaxPoolSize(5);
// 设置队列容量
executor.setQueueCapacity(99);
// 设置线程活跃时间(秒)
executor.setKeepAliveSeconds(300);
// 设置默认线程名称
executor.setThreadNamePrefix("thread-");
// 设置拒绝策略rejection-policy:当pool已经达到max size的时候,如何处理新任务 CALLER_RUNS:不在新线程中执行任务,而是有调用者所在的线程来执行
executor.setRejectedExecutionHandler(new ThreadPoolExecutor.CallerRunsPolicy());
// 等待所有任务结束后再关闭线程池
executor.setWaitForTasksToCompleteOnShutdown(true);
return executor;
}
@Override
public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {
return new SimpleAsyncUncaughtExceptionHandler();
}
}
之后直接在需要调用的方法上使用注解@Async即可
需要注意的点:在调用的类上面需要使用注解@Component,交给Spring容器管理。
遇到的坑:因为楼主是使用junit测试,所以出现一个情况就是,主线程跑完了,异步线程还没跑完。没异常,但是一直没有执行成功。
原因:因为主线程跑完了,异步线程也被终结掉了。我傻了。居然是这么简单的问题。。。
后面发现,如果想要使用junit测试,主线程不终止的情况,则直接死循环即可,while(true){};
因为spring那边也是这样去处理
记录之。