1.线程池基础
(1)为什么要使用线程池?
-
降低资源消耗:重复利用已经创建的线程,降低线程创建和销毁的消耗。
-
提升系统响应时间:当有任务时不需要等待线程的创建任务就能执行。
-
提高线程的可管理性:线程池可以统一分配,调优和管理。
(2)什么是线程池?
基于池化思想管理线程的工具。将线程事先创建好,交给线程池管理,当有任务时,分配线程池中的线程执行任务。当任务执行完成时可以选择关闭线程池或者继续等待任务。
2.创建线程池的方式(8种)
阿里代码规范:线程池不允许Excutors创建,而是通过ThreadPoolExecutor的方式。
(1)Excutors的方式创建线程,FixedThreadPool和SingleThreadPool,允许请求队列长度为Integer.MAX_VALUE,可能会堆积大量的请求,可能导致OOM。
(2)CachedThread运行的创建线程数为Integer.MAX_VALUE,可能会创建大量线程,导致OOM。
ThreadPoolExecutor构造方法参数
核心线程:只要线程池没有关闭就一直存在。
非核心线程:在非核心线程是空闲线程时,超过最大存活时间就会并销毁。
任务队列:交给线程池的任务(链式阻塞队列、数组阻塞队列)
线程工厂:实现newThread方法,自定义线程。
任务拒绝策略:线程池中的线程已满+无法继续扩容+没有空闲线程+任务队列已满
(1)原生的方法创建线程池
public class CustomThreadFactory implements ThreadFactory {
/**
* 计数器
*/
private final AtomicInteger i=new AtomicInteger(1);
@Override
public Thread newThread(Runnable r) {
Thread thread=new Thread(r);
thread.setName("线程"+i.getAndIncrement()+"号");
return thread;
}
}
public static void main(String[] args) {
ThreadPoolExecutor threadPoolExecutor=new ThreadPoolExecutor(10,25,10L, TimeUnit.SECONDS,
new LinkedBlockingDeque<>(),new CustomThreadFactory(),new ThreadPoolExecutor.AbortPolicy());
}
3.execute和submit的区别
-
runnable任务:无返回值,execute提交给线程池。
如果用submit提交给线程池,返回值null、或者指定返回值。
-
callable任务:有返回值,callable任务将返回结果封装到future对象中。通过对future对象的处理得到返回值。
execute | submit | |
---|---|---|
继承 | Executor | ExecutorService |
提交任务类型 | Runnable | Runnable、Callable |
返回值 | void | Future |
4.从Future对象获取结果
(1)get()方法:阻塞方法,获取任务执行结果的线程会一直等到任务执行完毕。
可能的异常:任务被取消、任务抛出异常、执行任务的线程中断
(2)get(int timeout,TimeUnit unit) :阻塞方法
可能的异常:任务被取消、任务抛出异常、执行任务的线程中断、等待任务执行超时
5.取消线程
cannel取消任务,参数是布尔类型传递ture中断线程。这个线程只对执行中的任务有意义。
状态:提交任务----->放入任务队列------->任务完成
取消未执行的任务(任务队列中)
取消已完成的任务:无法被取消
取消正在执行的任务
- 传参false,任务取消成功,但是依然会执行完。调用get方法会引发任务取消异常。
- 传参true,任务取消成功,响应中断线程指令或任务继续执行完。
6.拒绝策略
discard policy直接拒绝
caller run policy使用调用者线程执行被拒绝的线程。
abort policy默认拒绝策略、抛出RejectedExecutionException异常
discard old policy丢弃原任务队列的头部任务,然后添加被拒绝的任务。
7.shutdown关闭线程池
不会再接收任务、执行完已经提交的任务
与shutdownnow的区别?
shutdownnow方法,立刻停止线程池中执行的任务。返回任务队列中的线程,开发者自行处理没有执行的任务。
8.线程池的生命周期
-
running:正在执行任务并且可以接收新任务
-
shutdown:线程池准备关闭,不接受新任务但可以执行任务队列的任务
-
stop:线程池准备关闭,不接受新任务,终止池中任务,不处理任务队列的任务
-
tidying:所有任务已终止,线程池线程数为0;
-
terminated:彻底关闭线程池
9.面试题:线程池是怎么执行任务的
execute方法:
-
判断核心线程数是否已满
-没满,添加线程,结束。
-已满往下执行。
-
判断任务队列是否已满
-任务队列没满,任务添加到任务队列。
添加到任务队列后,判断线程池是否还在运行。没有就拒绝。线程池如果还在运行就判断还有可用线程吗,没有就创建非核心线程。
-任务队列已满,继续往下执行
-
判断线程池中的线程数是否已满
-没满就分配线程执行任务。
-已满就拒绝掉任务。
10.线程池处理批量任务
线程池.invokeall()方法,执行批量任务。这些任务只能是实现callable接口的有返回值的任务。
也可以指定时间,时间到了没完成的任务全部取消。
invokeany()方法,一旦有任务直接完,其他任务都不执行了,返回最先完成任务的返回值。
11.延时只执行一次的任务
Scheduled Thread Pool Executor延时线程池
线程池.schedule(任务,时间,单位)//提交任务后延时执行
12.周期、重复的任务
scheduleAtFixedRate //每1秒执行一次任务
scheduleWithFixedDelay //任务结束1s后再次执行
13.线程池监控