探索JAVA并发 - 线程池详解
2019-08-01 11:47
357 查看
作者:acupt,80后资深Java工程师一枚!架构师社区合伙人!
线程池是并发编程中必不可少的一种工具,也是面试高频话题。
- 线程复用:线程池中的线程是可以复用的,省去了创建、销毁线程的开销,提高了资源利用率(创建、销毁等操作都是要消耗系统资源的)和响应速度(任务提交过来线程已存在就不用等待线程创建了);
- 合理利用资源:通过调整线程池大小,让所有处理器尽量保持忙碌,又能防止过多线程产生过多竞争浪费资源;
Executor框架
在JAVA中,任务执行的主要抽象不是Thread,而是Executor。Executor基于生产者-消费者模式,提交任务的操作相当于生产者,执行任务的线程相当于消费者。所谓Executor框架,其实就是定义了一个接口,我们常用的线程池ThreadPoolExecutor就是对这个接口的一种实现。
Executors与常用线程池
Executors其实就是Executor(加s)Executors是一个Executor的工厂,有很多定义好的工厂方法,可以帮助懒惰的开发者快速创建一个线程池。下面是几个常用的工厂方法:
- newFixedThreadPool 固定长度线程池,每次提交任务都会创建一个新线程,直到线程数量达到指定阈值则不再创建新的;
- newCachedThreadPool 可缓存线程池,每次提交任务都会创建一个新线程(理论上无限制),部分任务执行完后如果没有新的任务,导致某些线程无用武之地,它们将被终结;
- newSingleThreadExecutor 只有一个线程的线程池;
- newScheduledThreadPool 可以延时或者定时执行任务的线程池。
线程池构造参数
- int corePoolSize : 核心线程数,有新任务来时,如果当前线程小于核心线程,则新建一个线程来执行该任务
- int maximumPoolSize : 最大线程数,线程池最多拥有的线程数
- long keepAliveTime : 空闲线程存活时间
- TimeUnit unit : 空闲线程存活时间的单位
- BlockingQueue workQueue : 存放待执行任务的阻塞队列,新任务来时,若当前线程数>=最大核心线程数,则放到这个队列
- ThreadFactory threadFactory : 创建新线程的工厂,一般用来给线程取个名字方便排查问题
- RejectedExecutionHandler handler : 任务被拒绝后的处理器,默认的处理器会直接抛出异常,建议重新实现
线程池创建示例
有返回值的提交方式
submit
ThreadPoolExecutor.execute()方法是没有返回值的,也就是说把任务提交给线程池后,我们就失去了它的消息,除非你还保留着它的引用,并且在里面有维护状态。如果不想这么麻烦,可以使用ThreadPoolExecutor.submit()来提交任务,这个方法会返回一个Future对象,通过这个对象可以知道任务何时被执行完。submit Callable
如果想任务有返回值,可以使用Callable作为任务定义。submit实现原理
为什么submit就可以让用户等待、获取任务返回?从源码讲起:关闭线程池
为什么需要关闭线程池?- 如果线程池里的线程一直存活,而且这些线程又不是守护线程,那么会导致虚拟机无法正常退出;
- 如果直接粗暴地结束应用,线程池中的任务可能没执行完,业务将处于未知状态;
- 线程中有些该释放的资源没有被释放。
- shutdown 停止接收新任务(继续提交会被拒绝,执行拒绝策略),但已提交的任务会继续执行,全部完成后线程池彻底关闭;
- shutdownNow 立即停止线程池,并尝试终止正在进行的线程(通过中断),返回没执行的任务集合;
- awaitTermination 阻塞当前线程,直到全部任务执行完,或者等待超时,或者被中断。
扩展线程池
总结
举个例子:如果设置了核心线程 < 最大线程数不等(一般都这么设置),但是又设置了一个很大的阻塞队列,那么很可能只有几个核心线程在工作,普通线程一直没机会被创建,因为核心线程满了会优先放到队列里,而不是创建普通线程。
长按订阅更多精彩▼ 如有收获,点个在看,诚挚感谢
相关文章推荐
- Java并发:线程池详解(ThreadPoolExecutor)
- Java并发之线程池详解
- Java高并发之线程池详解
- Java并发核心基础——线程池使用及底层实现机制详解
- Java高并发之线程池详解
- Java并发编程 11 Executor线程池详解
- Java中List集合的列表迭代器详解(内附一个关于并发修改异常的小案例以及List的一道面试题)
- Java并发之线程池(一)
- Java并发——线程池原理
- java 线程池使用和详解
- [置顶] Java并发之ReentrantLock详解
- 【JDK并发包基础】线程池详解
- java 线程池 详解
- Java线程池详解
- Java线程池使用与原理详解
- Java并发之AQS详解
- Java并发编程:线程池的使用
- 深入学习java并发编程:线程池ThreadPoolExecutor实现以及使用
- 聊聊并发(三)――JAVA线程池的分析和使用
- 2011-09-06 17:58 Java并发编程:定制线程池的大小