每逢面试都会询问道线程池的概念和使用,但是工作中真正的又有多少场景使用呢?相信大家都会有这样的疑问:面试选拔造汽车,实际进公司就是拧螺丝!但是真正要把这颗螺丝拧紧,拧牢,没有这些最底层的知识做铺垫你可以吗?你能胜任吗?所以嘛,公司招聘还是有一些尺度和标准的,大家学好技术才是硬道理。好啦,接下来我们就谈谈线程池的使用和概念。
一、什么是线程池?
为了避免系统频繁地创建线程和销毁线程,我们可以让创建的线程进行复用。如果大家进行过数据库开发,对数据库连接池应该不会陌生,为了避免每次数据库查询都重新建立和销毁数据库连接,我们可以使用数据库连接池维护一些数据库连接,让他们长期保持在一个激活状态。当系统需要使用数据库时,并不是创建一个新的连接,而是从连接池中获得一个可用的连接即可。反之,当需要关闭连接时,并不真的把连接关闭,而是将这个连接 "还" 给连接池即可。通过这样的方式,可以节约不少创建和销毁对象的时间。
线程池也是类似的概念。线程池中,总有那么几个活跃线程。当你需要使用线程时,可以从池子中随便拿一个空闲线程,当完成工作时,并不急着关闭线程,而是将这个线程退回到池子,方便其他人使用!
二、JDK对线程池的支持,各个接口、类之间的关系,类图展示
其中 ThreadPoolExecutor 表示一个线程池。Executors 类则扮演着线程池工厂的角色,通过 Executors 可以获取一个拥有特定功能的线程池。从上图亦可知,ThreadPoolExecutor 类实现了Executor 接口,因此可以通过这个接口,任何 Runnable 的对象都可以被 ThreadPoolExecutor 线程池调度!
三、Executors 内部创建线程池详解
Executor 框架提供了各种类型的线程池,主要有以下工厂方法:
public static ExecutorService newFixedThreadPool(int nThreads) public static ExecutorService newFixedThreadPool(int nThreads, ThreadFactory threadFactory) public static ExecutorService newSingleThreadExecutor() public static ScheduledExecutorService newSingleThreadScheduledExecutor() public static ScheduledExecutorService newScheduledThreadPool(int corePoolSize, ThreadFactory threadFactory)
以上工厂方法分别返回具有不同工作特性的线程池。这些线程池工厂方法的具体说明如下:
1)newFixedThreadPool() 方法:该方法返回一个固定线程数量的线程池。该线程池中的线程数量始终不变。当有一个新的任务提交时,线程池中若有空闲线程,则立即执行。若没有,则新的任务会被暂存在一个任务队列中,待有线程空闲,便处理在任务队列中的任务。
2)newSingleThreadPool() 方法:该方法返回一个只有一个线程的线程池。若多余一个任务会提交到该线程池,任务会被保存在一个任务队列中,待线程空闲,按先入先出的顺序执行队列中的任务;
3)newCachedThreadPool() 方法:该方法返回一个可根据实际情况调整线程数量的线程池。线程池的线程数量不确定,但若有空闲线程可以复用,则会优先使用可复用的线程。若所有线程均在工作,又有新的任务提交,则会创建新的线程处理任务。所有线程在当前
任务执行完毕后,将返回线程池进行复用。
4)newSingleThreadScheduledExecutor() 方法:该方法返回一个 ScheduledExecutorService 对象,线程池大小为 1。ScheduledExecutorService 接口在 ExecutorService 接口之上扩展了在给定时间执行某任务的功能,如在某个固定的延时之后执行,或者周期性执行某个任务。
5)newScheduledThreadPool() 方法:该方法也返回一个 ScheduledExecutorService 对象,但该线程池可以执行线程数量。
每一篇的篇幅不易过长,阅读起来会使人产生一种疲惫心里,下一篇:Java编发编程 - 线程池的认识(二)