目录
原理
好处
创建线程池
构造方法
参数详解
int corePoolSize
int maximumPoolSize
long keepAliveTime
TimeUnit unit
BlockingQueue workQueue
ThreadFactory threadFactory
RejectedExecutionHandler hanler
执行任务
execute()
submit()
并发操作
关闭线程池
shutdown()
shutdownNow()
原理
线程池就是一个存放已经创建好的线程的池子,当有任务提交给线程池执行时,线程池中的某个线程会主动执行该任务。如果线程池中的线程数量不够应付数量众多的任务时,需要自动扩充新的线程到线程池中,数量是有限的。当任务比较少的时候,线程池中的线程能够自动回收,释放资源。
好处
降低资源消耗:通过重复利用已创建的线程降低线程创建和销毁造成的消耗。
提高响应速度:当任务到达时,任务可以不需要等到线程创建就能立即执行。
提高线程的可管理性:线程是稀缺资源,如果无限制地创建,不仅会消耗系统资源,还会降低系统的稳定性,使用线程池可以进行统一分配、调优和监控。
创建线程池
通过java.util.concurrent.ThreadPoolExecutor来创建一个线程池。
构造方法
//五个参数的构造函数
public ThreadPoolExecutor(int corePoolSize,
int maximumPoolSize,
long keepAliveTime,
TimeUnit unit,
BlockingQueue<Runnable> workQueue)
//六个参数的构造函数-1
public ThreadPoolExecutor(int corePoolSize,
int maximumPoolSize,
long keepAliveTime,
TimeUnit unit,
BlockingQueue<Runnable> workQueue,
ThreadFactory threadFactory)
//六个参数的构造函数-2
public ThreadPoolExecutor(int corePoolSize,
int maximumPoolSize,
long keepAliveTime,
TimeUnit unit,
BlockingQueue<Runnable> workQueue,
RejectedExecutionHandler handler)
//七个参数的构造函数
public ThreadPoolExecutor(int corePoolSize,
int maximumPoolSize,
long keepAliveTime,
TimeUnit unit,
BlockingQueue<Runnable> workQueue,
ThreadFactory threadFactory,
RejectedExecutionHandler handler)
参数详解
int corePoolSize
该线程池中核心线程的最大数量,即默认情况下一直会存活在线程池中,即使处于闲置状态。如果当前线程总数小于corePoolSize,新建的线程是核心线程,否则是非核心线程。
int maximumPoolSize
该线程池中线程总数最大值。
long keepAliveTime
表示线程没有任务执行时最多保持多久时间会终止。默认情况下,只有当线程池中的线程数大于corePoolSize时,keepAliveTime才会起作用,即当线程池中的线程数大于corePoolSize时,如果一个线程空闲的时间达到keepAliveTime,则会终止,直到线程池中的线程数不超过corePoolSize。但是如果调用了allowCoreThreadTimeOut(boolean)方法,在线程池中的线程数不大于corePoolSize时,keepAliveTime参数也会起作用,直到线程池中的线程数为0。
TimeUnit unit
参数keepAliveTime的时间单位
TimeUnit是一个枚举类型,其包括:
NANOSECONDS : 1微毫秒 = 1微秒 / 1000
MICROSECONDS : 1微秒 = 1毫秒 / 1000
MILLISECONDS : 1毫秒 = 1秒 /1000
SECONDS : 秒
MINUTES : 分
HOURS : 小时
DAYS : 天
BlockingQueue<Runnable> workQueue
该线程池中的任务队列,用来存储等待执行的任务。
ThreadFactory threadFactory
线程工厂,主要用来创建线程,一般用不上。
RejectedExecutionHandler hanler
表示当拒绝处理任务时的策略,有以下四种取值:
ThreadPoolExecutor.AbortPolicy:丢弃任务并抛出RejectedExecutionException异常。
ThreadPoolExecutor.DiscardPolicy:也是丢弃任务,但是不抛出异常。
ThreadPoolExecutor.DiscardOldestPolicy:丢弃队列最前面的任务,然后重新尝试执行任务(重复此过程)
ThreadPoolExecutor.CallerRunsPolicy:由调用线程处理该任务
执行任务
execute()
这个方法是顶层接口Executor中声明的方法,在ThreadPoolExecutor类中进行了具体实现,参数是Runnable,通过这个方法可以向线程池提交一个任务,交由线程池去执行。
submit()
Submit是在接口ExecutorService中声明的方法,在ThreadPoolExecutor类的抽象父类AbstractExecutorService中进行了具体实现。这个方法也是用来向线程池提交任务的,参数是Runnable,但是它和execute()方法不同,它能够返回任务执行的结果,去看submit()方法的实现,会发现它实际上还是调用的execute()方法,只不过它利用了Future来获取任务执行结果。
并发操作
可以和CountDownLatch配合使用:
1、主线程使用await() 阻塞住,直到所有子线程都执行完毕了,才会继续向下执行。
2、所有子线程共享同一个 CountDownLatch变量,需要执行的线程总数。
3、所有子线程的finally块中,必须要countDown() ,确保 "无论 正确、异常 都会countDown",否则主线程会由于"某一个 子任务 没有countDown过,就执行结束了,导致CountDownLatch最终无法被countDown 到 0 ,而被永远挂住。
关闭线程池
shutdown()
不会立即终止线程池,而是要等所有任务缓存队列中的任务都执行完后才终止,但再也不会接受新的任务。
shutdownNow()
立即终止线程池,并尝试打断正在执行的任务,并且清空任务缓存队列,返回尚未执行的任务。