到底如何设置 Java 线程池的大小?

0 前言

在我们日常业务开发过程中,或多或少都会用到并发的功能。那么在用到并发功能的过程中,就肯定会碰到下面这个问题 

并发线程池到底设置多大呢? 

通常有点年纪的程序员或许都听说这样一个说法 (其中 N 代表 CPU 的个数)

1.CPU密集型应用,线程池大小设置为N+ 12.IO密集型应用,线程池大小设置为 2N

1 这个说法到底是不是正确的呢?

其实这是极不正确的。那为什么呢? 

首先我们从反面来看,假设这个说法是成立的,那我们在一台服务器上部署多少个服务都无所谓了。因为线程池的大小只能服务器的核数有关,所以这个说法是不正确的。那具体应该怎么设置大小呢? 

假设这个应用是两者混合型的,其中任务即有 CPU 密集,也有 IO 密集型的,那么我们改怎么设置呢?是不是只能抛硬币来决定呢? 

那么我们到底该怎么设置线程池大小呢?有没有一些具体实践方法来指导大家落地呢?让我们来深入地了解一下。

Little's Law(利特尔法则)

一个系统请求数等于请求的到达率与平均每个单独请求花费的时间之乘积


假设服务器单核的,对应业务需要保证请求量(QPS):10,真正处理一个请求需要 1 秒,那么服务器每个时刻都有 10 个请求在处理,即需要 10 个线程


同样,我们可以使用利特尔法则(Little’s law)来判定线程池大小。我们只需计算请求到达率和请求处理的平均时间。然后,将上述值放到利特尔法则(Little’s law)就可以算出系统平均请求数。估算公式如下:

*线程池大小 = ((线程 IOtime+ 线程 CPUtime)/线程 CPUtime) CPU数目**

2 具体实践

通过公式,我们了解到需要 3 个具体数值:

一个请求所消耗的时间 (线程 IOtime+ 线程 CPUtime)该请求计算时间 (线程 CPUtime)CPU 数目

3 请求消耗时间

Web 服务容器中,可以通过 Filter 来拦截获取该请求前后消耗的时间

publicclassMoniterFilterimplementsFilter{privatestaticfinalLogger logger = LoggerFactory.getLogger(MoniterFilter.class);@OverridepublicvoiddoFilter(ServletRequest request, ServletResponse response, FilterChain chain)throwsIOException,            ServletException {longstart = System.currentTimeMillis();        HttpServletRequest httpRequest = (HttpServletRequest) request;        HttpServletResponse httpResponse = (HttpServletResponse) response;        String uri = httpRequest.getRequestURI();        String params = getQueryString(httpRequest);try{            chain.doFilter(httpRequest, httpResponse);}finally{longcost = System.currentTimeMillis() - start;logger.info("access url [{}{}], cost time [{}] ms )", uri, params, cost);        }privateStringgetQueryString(HttpServletRequest req){StringBuilder buffer =newStringBuilder("?");        Enumeration<String> emParams = req.getParameterNames();try{while(emParams.hasMoreElements()) {                String sParam = emParams.nextElement();                String sValues = req.getParameter(sParam);buffer.append(sParam).append("=").append(sValues).append("&");            }returnbuffer.substring(0, buffer.length() -1);}catch(Exception e) {logger.error("get post arguments error", buffer.toString());        }return"";    }}

4 CPU计算时间

CPU计算时间 = 请求总耗时 - CPU IO time

假设该请求有一个查询 DB 的操作,只要知道这个查询 DB 的耗时(CPU IO time),计算的时间不就出来了嘛,我们看一下怎么才能简洁,明了的记录 DB 查询的耗时。

通过(JDK 动态代理/ CGLIB)的方式添加 AOP 切面,来获取线程 IO 耗时。代码如下,请参考:

publicclassDaoInterceptorimplementsMethodInterceptor{privatestaticfinalLogger logger = LoggerFactory.getLogger(DaoInterceptor.class);@OverridepublicObjectinvoke(MethodInvocation invocation)throwsThrowable{StopWatch watch =newStopWatch();        watch.start();Object result =null;Throwable t =null;try{            result = invocation.proceed();}catch(Throwable e) {t = e ==null?null: e.getCause();throwe;}finally{            watch.stop();logger.info("({}ms)", watch.getTotalTimeMillis());        }returnresult;    }}

5 CPU数目

逻辑 CPU 个数 ,设置线程池大小的时候参考的 CPU 个数 

cat/proc/cpuinfo| grep"processor"| wc -lSSSS

6 总结

合适的配置线程池大小其实很不容易,但是通过上述的公式和具体代码,我们就能快速、落地的算出这个线程池该设置的多大。

不过最后的最后,我们还是需要通过压力测试来进行微调,只有经过压测测试的检验,我们才能最终保证的配置大小是准确的。


文章参考:https://mp.weixin.qq.com/s/Mnod-XP9bcyyb11Q83hjeQ

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,544评论 6 501
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,430评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,764评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,193评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,216评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,182评论 1 299
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,063评论 3 418
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,917评论 0 274
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,329评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,543评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,722评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,425评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 41,019评论 3 326
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,671评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,825评论 1 269
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,729评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,614评论 2 353

推荐阅读更多精彩内容