线程池 Executors
在之后的Java开发中,我们会不可避免的创建大量线程,线程完成任务后就被销毁了,这种情况下,线程执行任务的流程是创建线程,创建任务,执行任务,关闭线程,而创建和释放线程会占据整个流程的多数时间。如果并发的线程数量很多,并且每个线程都是执行一个时间很短的任务就结束了,这样频繁创建线程 就会大大降低 系统的效率,因为频繁创建线程和销毁线程需要时间. 线程池就是一个容纳多个线程的容 器,池中的线程可以反复使用,省去了频繁创建线程对象的操作,节省了大量的时间和资源。
一个线程池包括以下四个基本组成部分:
1、线程池管理器(ThreadPool):用于创建并管理线程池,包括 创建线程池,销毁线程池,添加新任务;
2、工作线程(PoolWorker):线程池中线程,在没有任务时处于等待状态,可以循环的执行任务;
3、任务接口(Task):每个任务必须实现的接口,以供工作线程调度任务的执行,它主要规定了任务的入口,任务执行完后的收尾工作,任务的执行状态等;
4、任务队列(taskQueue):用于存放没有处理的任务。提供一种缓冲机制。
线程池的好处:
- 降低资源消耗。
- 提高响应速度。
- 提高线程的可管理性
Java中的四种线程池 . ExecutorService
缓存线程池
长度无限制
ExecutorService service = Executors.newCachedThreadPool();
执行流程:
- 判断线程池是否存在空闲线程
- 存在则使用
- 不存在,则创建线程 并放入线程池, 然后使用
应用场景:执行大量、耗时少的任务。
定长线程池
长度是指定的数值
ExecutorService service = Executors.newFixedThreadPool(2);
执行流程:
- 判断线程池是否存在空闲线程
- 存在则使用
- 不存在空闲线程,且线程池未满的情况下,则创建线程 并放入线程池, 然后使用
- 不存在空闲线程,且线程池已满的情况下,则等待线程池存在空闲线程
应用场景:控制线程最大并发数。
单线程线程池
效果与定长线程池创建时传入数值1效果一致
ExecutorService service = Executors.newSingleThreadExecutor();
执行流程:
- 判断线程池 的那个线程 是否空闲
- 空闲则使用
- 不空闲,则等待 池中的单个线程空闲后使用
应用场景:不适合并发但可能引起 IO 阻塞性及影响 UI 线程响应的操作,如数据库操作、文件操作等。
周期性任务定长线程池
执行流程:
- 判断线程池是否存在空闲线程
- 存在则使用
- 不存在空闲线程,且线程池未满的情况下,则创建线程 并放入线程池, 然后使用
- 不存在空闲线程,且线程池已满的情况下,则等待线程池存在空闲线程
周期性任务执行时:定时执行, 当某个时机触发时, 自动执行某任务
public static void main(String[] args) {
/**
* 定时执行
* 参数1. runnable类型的任务
* 参数2. 时长数字
* 参数3. 时长数字的单位 */
ScheduledExecutorService service = Executors.newScheduledThreadPool(1);
service.schedule(new Runnable() {
@Override public void run() {
System.out.println("周期一");
}
},5,TimeUnit.SECONDS);
/**
* 周期执行
* 参数1. runnable类型的任务
* 参数2. 时长数字(延迟执行的时长)
* 参数3. 周期时长(每次执行的间隔时间)
* 参数4. 时长数字的单位 */
ScheduledExecutorService service = Executors.newScheduledThreadPool(2);
service.scheduleAtFixedRate(new Runnable() {
@Override public void run() {
System.out.println("周期二");
}
},5,2,TimeUnit.SECONDS);
}
应用场景:执行定时或周期性的任务。