美文网首页JavaJava 杂谈程序员专栏
工作多年,关于线程数存在的误区

工作多年,关于线程数存在的误区

作者: 享学课堂 | 来源:发表于2019-05-16 12:21 被阅读1次

享学课堂特邀作者:老顾
转载请声明出处!

前言

“不好了,线上服务器超时严重,请求非常慢,好像报连接数too many了,怎么办?“小伙伴们在反馈。一般我们的技术老大的处理方式,把连接数和线程池调大点,重启,在观察。
往往这个方式是应急措施,治标不治本,因为不知道问题的原因
有个严重误区,以为线程池设置太小了,调大点请求就会快了。
今天老顾就带着小伙伴们沟通一下,线程池的大小应该如何合理的设置其大小?

问题

如果有两个任务需要处理,一个任务A,一个任务B

方案一:一个线程执行任务A和B,A执行完后,执行B
方案二:两个线程A和B去执行任务A 和 B,同时进行

哪个方案会快点?应该很多人会回答,肯定是方案二啊,多线程并行去处理任务A和B,肯定快啊。是这样吗?回答这个问题之前,先带着大家去回顾梳理一下。

线程执行

多线程的执行,是由CPU进行调度的,一个CPU在同一时刻只会执行一个线程,我们看上去的线程A 和 线程B并发执行,为了让用户感觉这些任务正在同时进行,操作系统利用了时间片轮转的方式,CPU给每个任务都服务一定的时间,然后把当前任务的状态保存下来,在加载下一任务的状态后,继续服务下一任务。任务的状态保存及再加载,这段过程就叫做上下文切换

上下文切换的过程是需要时间的;现在我们再来看一下上面的问题,我们小伙伴们再看一下是哪个方案快呢?是不是有些小伙伴们会说方案一,因为不需要线程切换。先不急,再往下看

为什么要多线程

按照上面的说法,有线程的上下文切换耗时,那么为什么需要多线程呢?小伙伴会不会感觉很乱,怎么回事?
小伙伴想想在我们真实业务中,我们是什么流程?

上图的流程:

1、先发起网络请求
2、web服务器解析请求
3、请求后端的数据库获取数据
4、获取数据后,进行处理
5、把处理结果放回给用户

这个是我们处理业务的时候,常规的请求流程;我们看一下整个过程涉及到什么计算机处理。

1、网络请求----->网络IO
2、解析请求----->CPU
3、请求数据库----->网络IO
4、mysql查询数据----->磁盘IO
5、mysql返回数据----->网络IO
6、数据处理----->CPU
7、返回数据给用户----->网络IO

小伙伴们是不是感觉又不乱了,在真实业务中我们不单单会涉及cpu计算,还有网络IO和磁盘IO处理,这些处理是非常耗时的。如果一个线程整个流程是上图的流程,真正涉及到CPU的只有2个节点,其他的节点都是IO处理,那么线程在做IO处理的时候,CPU就空闲出来了,CPU的利用率就不高

小伙伴们现在知道多线程的用处了吧,对,就是为了提升CPU利用率

提升QPS/TPS

衡量系统性能如何,主要指标系统的(QPS/TPS)。

QPS/TPS:每秒能够处理请求/事务的数量
并发数:系统同时处理的请求/事务的数量
响应时间:就是平均处理一个请求/事务需要时长

QPS/TPS = 并发数/响应时间

也就是并发数越大,QPS就越大;所以很多人就会以为调大线程池,并发数就会大,也会提升QPS,所以才会出现一开始前言所说的。其实QPS还跟响应时间成反比,响应时间越大,QPS就会越小。

虽然并发数调大了,就会提升QPS,但线程数也会影响响应时间,因为上面我们也提到了上下文切换的问题,那怎么设置线程数的呢?

如何设置线程数

那我们如何分配线程?我们提供一个公式:

最佳线程数目 = ((线程等待时间+线程CPU时间)/线程CPU时间 )* CPU数目

备注这个公式也是前辈们分享的,当然老顾看了淘宝前台系统优化实践的文章,和上面的公式很类似,不过在CPU数目那边,他们更细化了,不过不管什么公式,最终还是在生产环境中运行后,再优化调整。

我们继续上面的任务,我们的服务器CPU核数为4核,一个任务线程cpu耗时为20ms,线程等待(网络IO、磁盘IO)耗时80ms,那最佳线程数目:( 80 + 20 )/20 * 4 = 20。也就是设置20个线程数最佳。

从这个公式上面我们就得出,线程的等待时间越大,线程数就要设置越大,这个正好符合我们上面的分析,充分利用cpu利用率。那从另一个角度上面说,线程数设置多大,是根据我们自身的业务的,需要自己去压力测试,设置一个合理的数值。

基础常规标准

那我们小伙伴们会问,因为很多业务集中到一个线程池中,不像上面的案例比较简单,事实上业务太多,怎么设置呢?这个就是要去压力测试去调整。不过我们的前辈已经帮我们总结了一个基础的值(最终还是要看运行情况自行调整)

CPU密集型:操作内存处理的业务,一般线程数设置为:CPU核数 + 1或者 CPU数*2。数为4的话,一般设置 5 或 8
IO密集型:文件操作,网络操作,数据库操作,一般线程设置为:cpu数 / (1-0.9),数为4的话,一般设置 40

总结

今天介绍了线程数大小的设置,一些小伙伴们的误区。讲到这里我们小伙伴们是不是对线程有了更新的理解,不像之前那么粗暴,应该要去分析为什么这么慢,系统的瓶颈出现在什么地方,减少瓶颈的耗时。

老顾可以推荐小伙伴们再去看一下redis、nginx;为什么他们会那么快呢?其实和这篇文章的知识点有共同的地方。谢谢阅读!!!

相关文章

  • 工作多年,关于线程数存在的误区

    享学课堂特邀作者:老顾转载请声明出处! 前言 “不好了,线上服务器超时严重,请求非常慢,好像报连接数too man...

  • Java开发多年经验分享,关于线程数存在的误区!

    前言“不好了,线上服务器超时严重,请求非常慢,好像报连接数too many了,怎么办?“小伙伴们在反馈。一般我们的...

  • ThreadPoolTaskExecutor小记

    corePoolSize(核心线程数)核心线程会一直存在,即使没有任务执行;当线程数小于核心线程数的时候,即使有空...

  • 详解IIS最大并发连接数

    最大并发连接数=队列长度+工作线程数 【工作线程数】 IIS实际可以第一时间处理的请求数。比如,工作线程数=100...

  • Tomcat 请求线程的解析

    查询tomcat当前并发连接数 关于tomcat 请求线程的问题 Tomcat 请求线程数

  • Java线程池

    线程池工作流程 核心线程数 2.最大线程数 3.工作队列当我们将任务丢给线程池时首先检查线程数有没有达到了核心线程...

  • Java线程池(ThreadPoolExecutor)构造参数

    线程池构造参数 corePoolSize: 核心线程数,即使闲置也会存在于线程池中,除非设置了allowCoreT...

  • netty

    工作线程数接收线程数 后台开发人员把工作线程数增加到150,对客户端的请求处理能力,增加了十几倍。 现在我们的Ap...

  • 线程数究竟设多少合理

    分享一篇,关于线程的经典文章。 一、需求缘起 Web-Server通常有个配置,最大工作线程数,后端服务一般也有个...

  • 关于多线程的理解

    1.关于多线程存在的意义: 首先,既然有多线程,存在既有意义,那么为什么存在多线程呢,多线程的用处在何处呢我的理解...

网友评论

    本文标题:工作多年,关于线程数存在的误区

    本文链接:https://www.haomeiwen.com/subject/ibobaqtx.html