Spring boot使用多线程过程步骤解析
Spring中实现多线程,其实非常简单,只需要在配置类中添加@EnableAsync就可以使用多线程。在希望执行的并发方法中使用@Async就可以定义一个线程任务。通过spring给我们提供的ThreadPoolTaskExecutor就可以使用线程池。
第一步,先在SpringBoot主类中定义一个线程池,比如:
packagecom.jmxf.core.config; importjava.util.concurrent.Executor; importorg.springframework.context.annotation.Bean; importorg.springframework.context.annotation.Configuration; importorg.springframework.scheduling.annotation.EnableAsync; importorg.springframework.scheduling.concurrent.ThreadPoolTaskExecutor; @Configuration @EnableAsync//启用异步任务 publicclassAsyncConfiguration{ //组件计算 @Bean("zjExecutor") publicExecutorasyncExecutor(){ ThreadPoolTaskExecutorexecutor=newThreadPoolTaskExecutor(); //核心线程数5:线程池创建时候初始化的线程数 executor.setCorePoolSize(5); //最大线程数5:线程池最大的线程数,只有在缓冲队列满了之后才会申请超过核心线程数的线程 executor.setMaxPoolSize(10); //缓冲队列500:用来缓冲执行任务的队列 executor.setQueueCapacity(500); //允许线程的空闲时间60秒:当超过了核心线程出之外的线程在空闲时间到达之后会被销毁 executor.setKeepAliveSeconds(60); //线程池名的前缀:设置好了之后可以方便我们定位处理任务所在的线程池 executor.setThreadNamePrefix("DailyAsync-"); executor.initialize(); returnexecutor; } }
有很多你可以配置的东西。默认情况下,使用SimpleAsyncTaskExecutor。
第二步,使用线程池
在定义了线程池之后,我们如何让异步调用的执行任务使用这个线程池中的资源来运行呢?方法非常简单,我们只需要在@Async注解中指定线程池名即可,比如:
packagecom.jmxf.service.fkqManage.zj; importorg.springframework.scheduling.annotation.Async; @Service publicclassCentreZj{ /** *多线程执行zj计算推数 *@paramfkqZj *@throwsException */ @Async("zjExecutor") publicCompletableFutureexecuteZj(FkqZjfkqZj){ if(fkqZj==null)return; Stringzjid=fkqZj.getZjid(); FkqHdzjdmzjdm=getZjdm(zjid); Stringzjlj=zjdm.getZjlj(); if(StringUtils.isBlank(zjlj))return; Objectbean=ApplicationContextProvider.getBean(zjlj); Methodmethod; try{ method=bean.getClass().getMethod("refresh",String.class); method.invoke(bean,zjid); }catch(Exceptione){ e.printStackTrace(); } } returnCompletableFuture.completedFuture(zjid); }
executeZj方法被标记为Spring的@Async注解,表示它将在一个单独的线程上运行。该方法的返回类型是CompleetableFuture而不是String,这是任何异步服务的要求。
第三步,调用测试
List>executeZjs=newArrayList<>(); for(FkqZjfkqZj:zjs){ CompletableFuture executeZj=centreZj.executeZj(fkqZj); executeZjs.add(executeZj); } //等待所以子线程结束后返回结果 for(CompletableFuture completableFuture:executeZjs){ CompletableFuture.allOf(completableFuture).join(); }
注意事项
异步方法和调用方法一定要写在不同的类中,如果写在一个类中,是没有效果的!
原因:
spring对@Transactional注解时也有类似问题,spring扫描时具有@Transactional注解方法的类时,是生成一个代理类,由代理类去开启关闭事务,而在同一个类中,方法调用是在类体内执行的,spring无法截获这个方法调用。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持毛票票。