前言:整理常见的五种常见线程池,线程池解决了手写线程的不受控风险,频繁创建开销大的问题(在java8中一个线程大概要消耗1M左右的内存)。但是在面对一些业务需求时我们往往不去使用线程池来创建线程,因为线程的一些核心参数不可控,并且某些线程池的工作队列无上限,会导致OOM。
思考:线程池简化了线程的创建,为什么阿里巴巴Java开发手册中强制要求线程池不允许使用Executors创建?
要想真正理解为什么线程池不允许使用Executors去创建,而是通过ThreadPoolExecutor的方式,首先要了解线程的几个核心参数和拒绝策略,本文不再阐述,下一篇将会详细介绍。
那么,让我们先来看看常见的五个线程池吧!
1.newFixedThreadPool
newFixedThreadPool是一个固定大小的线程池,它可以在初始化时设定线程池的大小,当线程池中的线程都在工作时,新的任务会被放置到一个队列中等待被执行,适用于一些稳定且较长的任务。
特征:
(1).可指定线程池的大小,能够较好的控制线程的并发量。
(2).线程可以重复被使用,在显示关闭之前,都将一直存在
(3).超出一定量的线程被提交时候需要在队列中等待
创建方式:
Executors.newFixedThreadPool(int nThreads);//nThreads为线程的数量
Executors.newFixedThreadPool(int nThreads,ThreadFactory threadFactory);//nThreads为线程的数量,threadFactory创建线程的工厂方式
2.newCachedThreadPool
介绍:
CachedThreadPool是一个大小可变的线程池,它可以根据实际需要创建新的线程,在当前线程池空闲时,可以为新任务创建的线程都将被回收。这种线程池一般适用于一些短且频繁的任务。
特征:
(1)线程池中数量没有固定,可达到最大值(Interger. MAX_VALUE)
(2)线程池中的线程可进行缓存重复利用和回收(回收默认时间为1分钟)
(3)当线程池中,没有可用线程,会重新创建一个线程
创建方式:
Executors.newCachedThreadPool();
3.newScheduledThreadPool
介绍:
newScheduledThreadPool是一个固定大小的线程池,它可以在指定的时间或延迟后执行任务,例如定时等,这种线程池适用于一些需要定时或延迟执行的任务。
特征:
(1)线程池中具有指定数量的线程,即便是空线程也将保留
(2)可定时或者延迟执行线程活动
(1)Executors.newScheduledThreadPool(int corePoolSize);// corePoolSize线程的个数
(2)Executors.newScheduledThreadPool(int corePoolSize, ThreadFactory threadFactory);// corePoolSize线程的个数,threadFactory创建线程的工厂
4.newSingleThreadExecutor
介绍:
newSingleThreadExecutor是一个只有一个线程的线程池,它可以保证任务的顺序执行,这种线程池适用于一些需要严格按照顺序依次执行的任务。
特征:
(1) 线程池中最多执行1个线程,之后提交的线程活动将会排在队列中以此执行
创建方式:
(1)Executors.newSingleThreadExecutor() ;
(2)Executors.newSingleThreadExecutor(ThreadFactory threadFactory);// threadFactory创建线程的工厂方式
5.newWorkStealingPool
newWorkSteallingPool是一种可以自适应地调整线程池大小的线程池,它会根据处理器核数创建相应数量的线程池。当其中一个线程正在执行一个任务时,如果该任务需要调用另一个任务,那么该线程会尝试窃取另一个线程队列中的人物来执行,这种线程池适用于一些需要任务之间相互协作的任务。
总结:
根据不同的需求,可以选择不同的线程池来优化并发编程的效率。FixedThreadPool和CachedThreadPool适合不同长度的任务,ScheduledThreadPool适合定时或延迟执行任务,SingleThreadExecutor适合需要执行的任务,WorkStealingPool适合任务之间需要相互协作的任务