一、为什么要使用线程池
我们知道线程的并发操作,并不是真正的同时执行,而是通过CPU的上下文切换来执行。因为CPU切换的速度很快,以至于我们感觉不到,会造成在一种在同一时间内执行了多个操作的错觉。
那么我们在使用多线程的时候,如果创建了大量的线程,就会造成CPU的频繁切换,反而导致效率降低,而线程的数量在显式创建线程时,其实是不可控的。另外,频繁的创建和销毁线程,也会造成较大的开销。所以,我们可以使用线程池来控制线程数量和减少线程频繁创建销毁的开销。
二、java中的四种线程池
- 单例线程池:Executors.newSingleThreadExecutor();
- 缓存线程池:Executors.newCachedThreadPool();
- 定长线程池:Executors.newFixedThreadPool(nThreads);
- 定时线程池:Executors.newScheduledThreadPool(corePoolSize);
1.单例线程池:
单例线程池,顾名思义就是线程池中只有一个工作线程执行任务。执行的是串行操作,按照顺序执行。
如果有其他请求任务时,此时线程没有空闲,就会将此任务放到队列中等待执行。那它的队列其实是一个无线大的队列,如果有大量的请求任务时,一个线程的执行速度是有限的,就有可能造成队列积压,甚至于导致OOM的问题。
2.缓存线程池:
这种线程池内部没有核心线程,在有新的任务的时候,如果存在空闲线程的话就用空闲线程来执行任务,如果没有空闲线程的话,就新建一个线程来执行任务,这个最大