前言

上文我们介绍了JDK中的线程池框架Executor。我们知道,只要需要创建线程的情况下,即使是在单线程模式下,我们也要尽量使用Executor。即:

ExecutorService fixedThreadPool = Executors.newFixedThreadPool(1);
//此处不该利用Executors工具类来初始化线程池

但是,在《阿里巴巴Java开发手册》中有一条

【强制】线程池不允许使用 Executors 去创建,而是通过 ThreadPoolExecutor 的方式,这样的处理方式让写的同学更加明确线程池的运行规则,规避资源耗尽的风险。
Executors 返回的线程池对象的弊端如下: FixedThreadPool 和 SingleThreadPool : 允许的请求队列长度为 Integer.MAX_VALUE,可能会堆积大量的请求,从而导致 OOM。 CachedThreadPool 和 ScheduledThreadPool : 允许的创建线程数量为 Integer.MAX_VALUE,可能会创建大量的线程,从而导致 OOM。

可以看到,这是一个强制性的规则,并且是不允许使用Executors来创建,建议使用ThreadPoolExecutor来创建线程池,那我们先来回顾一下ExecutorsThreadPoolExecutor

我们可以看到ThreadPoolExecutor已经是Executor的具体实现了,而且具有较多可配参数(可配参数见下方,可仅了解,用到时再进行详细查询)。Executors是一个创建线程池的工具类,查看其源码的话也会发现这几种创建线程池的方法也都是通过调用ThreadPoolExecutor来实现的。

ThreadPoolExecutor一共有四个构造函数,七个可配参数,分别是:

  1. corePoolSize: 线程池中保持存活线程的数量。
  2. maximumPoolSize: 线程池中允许线程数量的最大值
  3. keepAliveTime: 表示线程没有任务执行时最多保持多久时间会终止
  4. unit: 参数keepAliveTime的时间单位
  5. workQueue: 一个阻塞队列,用来存储等待执行的任务
  6. threadFactory: 线程工厂,主要用来创建线程
  7. handler:表示当拒绝处理任务时的策略

分析

那么Executors到底会导致什么问题,才会让开发手册中直接被定义为不允许了呢。首先就是一个血淋淋的教训,直接导致线上服务不可用,已经可以算是事故了。

实验

我们也可以现在我们本地进行一下小实验:

public class ExecutorsTesting {private static ExecutorService executor = Executors.newFixedThreadPool(15);public static void main(String[] args) {for (int i = 0; i < Integer.MAX_VALUE; i++) {executor.execute(new SubThread());}}
}class SubThread implements Runnable {@Overridepublic void run() {try {Thread.sleep(10000);} catch (InterruptedException e) {//do nothing}}
}

运行时指定JVM参数:-Xmx8m -Xms8m,大概几秒钟之后,会报出OOM错误:

Exception in thread "main" java.lang.OutOfMemoryError: GC overhead limit exceededat com.kaikeba.mybatis.ExecutorsTesting.main(ExecutorsTesting.java:10) //报错行数为上述代码中的executor.execute(new SubThread());

那么为什么会报出这个错误呢。

源码分析

我们先来看一下Executors中的FixedThreadPool是如何构造的。

public static ExecutorService newFixedThreadPool(int nThreads) {return new ThreadPoolExecutor(nThreads, nThreads,0L, TimeUnit.MILLISECONDS,new LinkedBlockingQueue<Runnable>());
}

可以看到对于存储等待执行的任务,FixedThreadPool是通过LinkedBlockingQueue来实现的。而我们知道LinkedBlockingQueue是一个链表实现的阻塞队列,而如果不设置其容量的话,将会是一个无边界的阻塞队列,最大长度为Integer.MAX_VALUE由于Executors中并未设置容量,所以应用可以不断向队列中添加任务,导致OOM错误

上面提到的问题主要体现在newFixedThreadPoolnewSingleThreadExecutor两个工厂方法上,并不是说newCachedThreadPoolnewScheduledThreadPool这两个方法就安全了,这两种方式创建的最大线程数可能是Integer.MAX_VALUE,而创建这么多线程,必然就有可能导致OOM。

如何该利用ThreadPoolExecutor来创建线程池呢?

我们其实可以看到Executors中的newFixedThreadPool其实也是调用ThreadPoolExecutor来实现的。正如手册中所说,当我们不用Executors默认创建线程池的方法,而直接自己手动去调用ThreadPoolExecutor,可以让写的同学更加明确线程池的运行规则,规避资源耗尽的风险。比如我们在Executors.newFixedThreadPool基础上给LinkedBlockingQueue加一个容量,当队列已经满了,而仍需要添加新的请求会抛出相应异常,我们可以根据异常做相应处理。

public static ExecutorService newFixedThreadPool(int nThreads) {return new ThreadPoolExecutor(nThreads, nThreads,0L, TimeUnit.MILLISECONDS,new LinkedBlockingQueue<Runnable>(10)); //添加容量大小
}

除了自己定义ThreadPoolExecutor外。还可以利用其它开源类库,如apache和guava等,可以有更多个性化配置。

参考文章:

https://www.hollischuang.com/archives/2888

https://stackoverflow.com/questions/1094867/when-should-we-use-javas-thread-over-executor#answer-34373289

https://blog.51cto.com/zero01/2306857

您的点赞与关注是对作者写作的最大的支持,谢谢!

欢迎各位关注个人公众号(关注后Java程序员必备Spring Mybatics以及其他框架的入门和拔高视频哦)

inputstreamreader未关闭会导致oom_【搞定面试官】你还在用Executors来创建线程池?会有什么问题呢?相关推荐

  1. 如何在一分钟内搞定面试官

    转载自   如何在一分钟内搞定面试官 很多人的求职面试的过程中都会遇到这个问题:  "请做个自我介绍." 有的人,可以口若悬河.妙语连珠讲3分钟,有的人,可能磕磕巴巴,讲了30秒, ...

  2. inputstreamreader未关闭会导致oom_Linux内核OOM机制分析和防止进程被OOM杀死的方法...

    问题描述 Linux 内核有个机制叫 OOM killer(Out-Of-Memory killer),该机制会监控那些占用内存过大,尤其是瞬间很快消耗大量内存的进程,为了防止内存耗尽而内核会把该进程 ...

  3. inputstreamreader未关闭会导致oom_ThreadLocal 一定会导致内存泄露?

    在面试的时候,ThreadLocal作为高并发常用工具经常会被问到.而面试官比较喜欢问的问题有以下两个: 1.ThreadLocal是怎么实现来保证每个线程的变量副本的. 2.ThreadLocal的 ...

  4. 「秘籍」JAVA求职面试宝典,轻松搞定面试官!拿下大厂offer

    小编收集了些面试相关应对技巧分享给大家 面试好比就像是场约会,你是不是常常担心自己没有表现好,又担心对方是个"渣男" 有时候吧, 你看上了对方,对方看不上你,还有的时候,你们互相看 ...

  5. 如何在一分钟内搞定面试官?

    很多人的求职面试的过程中都会遇到这个问题:  "请做个自我介绍." 有的人,可以口若悬河.妙语连珠讲3分钟,有的人,可能磕磕巴巴,讲了30秒,前者一定能胜过后者,然则未必,今天就来 ...

  6. 五年JAVA开发,一份简历搞定面试官!

    每年春节后两个月都是招聘高峰期,很多想换工作的职场人士都会选择在此时换一份工作,毕竟一年之计在于春,对于公司和个人而言都是一个新的开始. 大家在春节长假身心得到放松后,准备摩拳擦掌的找工作了,不过大部 ...

  7. 【Java面试】大厂裁员,小厂倒闭,如何搞定面试官Java SPI是什么?有什么用?

    "Java SPI是什么?有什么用?" 这是阿里p6面试过程中,第二面的时候遇到的一个真实的问题. 如果你不理解SPI,建议你看完整篇文章. 大家好,我是Mic,一个工作了14年的 ...

  8. 超详细的Spring Boot教程,搞定面试官!

    前言 Spring Boot是由Pivotal团队提供的全新框架,其设计目的是用来简化新Spring应用的初始搭建以及开发过程.该框架使用了特定的方式来进行配置,从而使开发人员不再需要定义样板化的配置 ...

  9. java 面试 框架_这份java集合框架面试题,轻松搞定面试官!

    集合框架 1. ArrayList 和 Vector 的区别. 这两个类都实现了 List 接口(List 接口继承了 Collection 接口),他们都是有序集 合,即存储在这两个集合中的元素的位 ...

最新文章

  1. keras 的 example 文件 cifar10_resnet.py 解析
  2. swift 之SnapKit 动画
  3. Oracle中限定日期,Oracle 日期的一些简单使用
  4. P7990-[USACO21DEC]Closest Cow Wins S【堆,贪心】
  5. Hadoop入门(二十二)Mapreduce的求平均值程序
  6. Effective C++学习第十天
  7. PHP+MySql+PDO实现简单增加、删除、修改、查询
  8. 团队开发-CodePlex的工作模式和团队协作开发
  9. 数据库管理工具 Navicat使用教程:导航窗格提示和技巧 - 管理连接
  10. Go基础-Go中的import
  11. a7100换电池_如何评价三星galaxy A7100(2016版)?
  12. 拍摄制作360度全景图有哪些技巧?
  13. 施工现场工地监管如何能够接入4G摄像头实现流媒体服务器视频监控?
  14. Arduino基础入门二之呼吸灯
  15. MNN实践[C++版本]
  16. nrm 切换不同的源工具
  17. unity中Camera的Field of View 垂直FOV
  18. godaddy新建二级域名
  19. sql语句批量导入数据库数据
  20. A站(ACFun)爬虫爬取并播放、下载视频(JAVA)

热门文章

  1. 面向对象的三大特性:封装、继承、多态
  2. DNS资源记录类型的总结
  3. edgesForExtendedLayout ios7新特性
  4. 关于Watir的upload file不能自动选择文件的解决方案
  5. gitlab+jenkins=自动化构建
  6. 本文实例讲解了PHP使用MySQL事物锁的实例,并备有注释加以详细说明
  7. 用JS实现版面拖拽效果
  8. Safari new Date()
  9. TS流解析 二 *****
  10. STL: string:erase