一、线程池
1. 为什么要使用线程池?
- 降低资源消耗。重复利用已创建的线程降低线程创建和销毁造成的消耗。
- 提高响应速度。当任务到达时,任务可以不需要等到线程创建就能立即执行。
- 提高线程的可管理性。线程时稀缺资源,如果无限制地创建,不仅会消耗系统资源,还会降低系统的稳定性,使用线程池可以进行统一分配、调优和监控。
2. 工作原理
当提交一个新任务到线程池时,线程池的处理流程如下:
-
线程池判断核心线程池里的线程是否都在执行任务。如果不是,则创建一个新的工作线程来执行任务。如果核心线程池里的线程都在执行任务,则进入下个流程。
-
线程池判断工作队列是否已经满了。如果工作队列没有满,则将新提交的任务存储在这个工作队列里。如果工作队列满了,则进入下个流程。
-
线程池判断线程池的线程是否都处于工作状态。如果没有,则创建一个新的工作线程来执行任务。如果已经满了,则交给饱和策略来处理这个任务。
3. 创建
ThreadPoolExecutor(int corePoolSize,
int maximumPoolSize,
long keepAliveTime,
TimeUnit unit,
BlockingQueue<Runnable> workQueue,
ThreadFactory threadFactory,
RejectedExecutionHandler handler);
参数说明:
- corePoolSize:线程池的基本大小。
- runnableTaskQueue:任务队列,用于保存等待执行的任务的阻塞队列。
- ArrayBlockingQueue:基于数组的有界阻塞队列,FIFO。
- LinkedBlockingQueue:基于链表的阻塞队列,FIFO,吞度量高于 ArrayBlockingQueue。
- SynchronousQueue:不存储元素的阻塞队列,每个插入操作必须等到另一个线程调用移除操作,否则插入操作一直处于阻塞状态。
- PriorityBlockingQueue:优先级无限阻塞队列。
- keepAliveTime:空闲线程存活时间。
- unit:keepAliveTime 指定的时间单位。
- maximumPoolSize:线程池最大数量,如果队列满了,并且已创建的线程数小于最大线程数,则线程池会再创建新的线程执行任务。若使用无界的任务队列这个参数无效。
- ThreadFactory:用于创建线程的工厂,可以通过线程工厂给每个创建出来的线程设置更有意义的名字。
- RejectedExecutionHandler:饱和策略,当队列和线程池都满了,说明线程池处于饱和状态,需要采取策略处理提交的新任务,默认时 AbortPolicy。
- AbortPolicy:直接抛出异常。
- CallerRunsPolicy:只用调用者所在的线程来执行任务。
- DiscardPolicy:不处理直接丢弃掉任务。
- DiscardOldestPolicy:丢弃掉阻塞队列中存放时间最久的任务,执行当前任务。
4. 提交任务
execute() 没有返回值,无法判断是否被线程池执行。
threadsPool.execute(new Runnable(){
@Override
public void run(){
}
});
submit() 返回 future 对象,通过 future 对象判断任务是否执行成功。
Future<Object> future = executor.submit(harReturnValuetask);
5. 关闭
shutdown() 或 shutdownNow(),遍历线程池的工作线程,然后逐个调用线程的 interrupt 方法来中断线程,所以无法响应中断的任务可能永远无法终止。
下篇《原则》