java線程池實(shí)戰(zhàn)應(yīng)用步驟詳解
一、線程池的創(chuàng)建方式
方式(一):通過(guò)構(gòu)造函數(shù)ThreadPoolExecutor()方式創(chuàng)建線程池
步驟1:先構(gòu)建線程池
public class AsyncTaskExecutor { /** * 核心線程數(shù) */ private static final int corePoolSize = 10; /** * 最大線程數(shù) */ private static final int maxPoolSize = 30; /** * 空閑線程回收時(shí)間 * 空閑線程是指:當(dāng)前線程池中超過(guò)了核心線程數(shù)之后,多余的空閑線程的數(shù)量 */ private static final int keepAliveTime = 100; /** * 任務(wù)隊(duì)列/阻塞隊(duì)列 */ private static final int blockingQueueSize = 99999; private static final ThreadPoolExecutor executorPool = new ThreadPoolExecutor( corePoolSize, maxPoolSize, keepAliveTime, TimeUnit.MILLISECONDS, new LinkedBlockingQueue<>(blockingQueueSize), new ThreadFactoryBuilder().setNameFormat("AsyncTaskThread" + "-%d").build(), new ThreadPoolExecutor.CallerRunsPolicy() ); /** * 異步任務(wù)執(zhí)行 * * @param task */ public static void execute(Runnable task) { executorPool.execute(task); } }
步驟2:通過(guò)實(shí)現(xiàn)Runnable接口,創(chuàng)建異步任務(wù)
@Slf4j public class CommonTask implements Runnable { /** * 模塊ID */ private Long modelId; /** * 模塊名稱 */ private ModelEnum modelName; /** * 構(gòu)造方法 * * @param modelId * @param modelName */ public CommonTask(Long modelId, ModelEnum modelName) { this.modelId = modelId; this.modelName = modelName; } @Override public void run() { log.info("start to process common task!!!"); if (modelId.intValue() == ModelEnum.Chinese.getCode()) { String name = ModelEnum.Chinese.getValue(); log.info("modelID = {} modelName = {}", modelId, name); } else { modelName = ModelEnum.forValue(modelId.intValue()); log.info("modelID = {} modelName = {}", modelId, modelName.getValue()); } } }
枚舉
public enum ModelEnum { Chinese(1, "語(yǔ)文"), Math(2, "數(shù)學(xué)"), English(3, "數(shù)學(xué)"); /** * code */ private int code; /** * value */ private String value; /** * 映射結(jié)果集 */ private static final Map<Integer, ModelEnum> VALUE_MAP; static { VALUE_MAP = new HashMap<>(); for (ModelEnum modelEnum : ModelEnum.values()) { VALUE_MAP.put(modelEnum.code, modelEnum); } } ModelEnum(int code, String value) { this.code = code; this.value = value; } public int getCode() { return code; } public String getValue() { return value; } /** * 根據(jù)code獲取枚舉實(shí)例 * * @param code * @return */ public static ModelEnum forValue(int code) { return VALUE_MAP.get(code); } }
步驟3:驗(yàn)證
//步驟1:創(chuàng)建異步任務(wù) CommonTask task = new CommonTask(1L, ModelEnum.Chinese); //步驟2:調(diào)用線程池異步執(zhí)行任務(wù) AsyncTaskExecutor.execute(task); log.info("main thread over...");
結(jié)果如下:
2024-05-23 14:53:16.096 INFO 20652 --- [ main] com.example.demo.dao.UserDaoTest : main thread over...
2024-05-23 14:53:16.097 INFO 20652 --- [yncTaskThread-0] com.example.demo.task.CommonTask : start to process common task!!!
2024-05-23 14:53:16.097 INFO 20652 --- [yncTaskThread-0] com.example.demo.task.CommonTask : modelID = 1 modelName = 語(yǔ)文
方式(二):構(gòu)建ThreadPoolTaskExecutor線程池,將其聲明為Bean,可以通過(guò)注入bean的方式和在方法上使用@Async(“asyncTaskExecutor”)這種注解方式使用此線程池
@Slf4j @Configuration @EnableAsync public class TaskExecutor { @Value("${async.executor.thread.core_pool_size}") private int corePoolSize; @Value("${async.executor.thread.max_pool_size}") private int maxPoolSize; @Value("${async.executor.thread.queue_capacity}") private int queueCapacity; @Value("${async.executor.thread.name.deal_task}") private String taskNamePrefix; /** * 構(gòu)建線程池 并將其聲明為Bean * 方式1:可以通過(guò)注入的方式使用此線程池 * 方式2:可以在方法上使用@Async("asyncTaskExecutor")這種注解方式使用此線程池 * * @return */ @Bean(name = "asyncTaskExecutor") public Executor asyncTaskExecutor() { log.info("start asyncTaskExecutor..."); ThreadPoolTaskExecutor threadPoolTaskExecutor = new ThreadPoolTaskExecutor(); threadPoolTaskExecutor.setCorePoolSize(corePoolSize); threadPoolTaskExecutor.setMaxPoolSize(maxPoolSize); threadPoolTaskExecutor.setQueueCapacity(queueCapacity); threadPoolTaskExecutor.setThreadNamePrefix(taskNamePrefix); //拒絕策略:當(dāng)前線程數(shù)已經(jīng)達(dá)到最大線程數(shù)后,如何處理新任務(wù) //CallerRunsPolicy()不在新線程中執(zhí)行任務(wù),而是返回調(diào)用者所在的線程來(lái)執(zhí)行 threadPoolTaskExecutor.setRejectedExecutionHandler( new ThreadPoolExecutor.CallerRunsPolicy()); threadPoolTaskExecutor.initialize(); return threadPoolTaskExecutor; } }
在application.yml中的配置
async: executor: thread: core_pool_size: 10 max_pool_size: 10 queue_capacity: 99999 name: deal_task: DEAL-TASK-
使用方式1:在方法上使用@Async(“asyncTaskExecutor”)這種注解方式使用此線程池
使用方式2:通過(guò)注入的方式使用此線程池
@Slf4j @Service public class WorkServiceImpl implements WorkService { @Async("asyncTaskExecutor") public void doAsyncTask() throws InterruptedException { Thread.sleep(1000); log.info("do AsyncTask..."); } }
@Slf4j @RunWith(SpringRunner.class) @SpringBootTest public class UserDaoTest { SqlSession sqlSession; @Resource private WorkService workService; @Resource private Executor asyncTaskExecutor; @Test public void test() { try { //方式1:在方法上使用@Async("asyncTaskExecutor")這種注解方式使用此線程池 for (int i = 0; i < 5; i++) { workService.doAsyncTask(); } log.info("main Thread over..."); //方式2:通過(guò)注入的方式使用此線程池(方便) asyncTaskExecutor.execute(() -> { log.info("async task is running.."); }); Thread.sleep(5000); }catch (Exception e) { e.printStackTrace(); } } }
到此這篇關(guān)于java線程池實(shí)戰(zhàn)應(yīng)用總結(jié)的文章就介紹到這了,更多相關(guān)java線程池內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Java使用集合實(shí)現(xiàn)斗地主分牌完整代碼
在斗地主游戲中,通常是將一副牌平均分成3份,每份17張牌,并留3張底牌,我們可以使用集合來(lái)實(shí)現(xiàn)這一功能,這篇文章主要給大家介紹了關(guān)于Java使用集合實(shí)現(xiàn)斗地主分牌的相關(guān)資料,需要的朋友可以參考下2024-05-05配置java.library.path加載庫(kù)文件問(wèn)題
這篇文章主要介紹了配置java.library.path加載庫(kù)文件問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-12-12EasyUi+Spring Data 實(shí)現(xiàn)按條件分頁(yè)查詢的實(shí)例代碼
這篇文章主要介紹了EasyUi+Spring Data 實(shí)現(xiàn)按條件分頁(yè)查詢的實(shí)例代碼,非常具有實(shí)用價(jià)值,需要的朋友可以參考下2017-07-07java實(shí)現(xiàn)的MD5摘要算法完整實(shí)例
這篇文章主要介紹了java實(shí)現(xiàn)的MD5摘要算法,結(jié)合完整實(shí)例形式分析了java實(shí)現(xiàn)md5單項(xiàng)加密的具體步驟與相關(guān)操作技巧,需要的朋友可以參考下2017-01-01Java中spring boot validation自定義注解使用方式
這篇文章主要介紹了Java中spring boot validation自定義注解使用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-08-08Mybatis報(bào)錯(cuò)mapkey is required問(wèn)題及解決
這篇文章主要介紹了Mybatis報(bào)錯(cuò)mapkey is required問(wèn)題及解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-06-06在win10系統(tǒng)下,如何配置Spring Cloud alibaba Seata以及出現(xiàn)問(wèn)題時(shí)怎么解決
今天教大家如何在win10系統(tǒng)下,配置Spring Cloud alibaba Seata以及出現(xiàn)問(wèn)題時(shí)怎么解決,文中有非常詳細(xì)的介紹及代碼示例,需要的朋友可以參考下2021-06-06SpringBoot?Webflux創(chuàng)建TCP/UDP?server并使用handler解析數(shù)據(jù)
這篇文章主要介紹了SpringBoot?Webflux創(chuàng)建TCP/UDP?server并使用handler解析數(shù)據(jù),具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-02-02MyBatis通用Mapper實(shí)現(xiàn)原理及相關(guān)內(nèi)容
今天小編就為大家分享一篇關(guān)于MyBatis通用Mapper實(shí)現(xiàn)原理及相關(guān)內(nèi)容,小編覺(jué)得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來(lái)看看吧2018-12-12