在Linux上查看活跃线程数与连接数

简介

现如今,有两种常见的软件资源几乎成了Java后端程序的标配,即线程池与连接池,但这些池化资源非常的重要,一旦不够用了,就会导致程序阻塞、性能低下,所以有时我们需要看看它们的使用情况,以判断这里是否是瓶颈。

查看活跃线程数

在Linux上,通过top -H -p 1命令,可以查看java进程的线程情况,其中1是java进程号,如下:
image_2023-03-04_20230304140014
如上,可以看到线程的名称、CPU使用率等,其中http-nio-8080-e就是Tomcat线程池中的线程,tomcat线程全名类似于http-nio-8080-exec-20,由于Linux中线程名称有长度限制,所以被截断了。

注:jdk8的话,需要jdk8u222以上版本,才能在top中看到线程名称。

我们数一下http-nio-8080-e线程的数量,发现它有20个,正好对应上了在springboot中的线程配置。
image_2023-03-04_20230304133328

这样能通过top得到线程池的线程数量了,但如何了解线程池的使用情况,即活跃线程有多少个呢?

经过查看man文档,我发现top命令有一个-i选项,描述如下:
image_2023-03-04_20230304134557
意思就是i是一个开关选项,默认会显示全部线程,而打开此选项之后,就只显示活跃线程了!

所以,只需要利用-i选项,再配合sed/awk/uniq等文本处理命令,即可以统计出活跃线程数了,如下:

$ top -H -i -b -d 1 -n2 -p 1 | awk -v RS= 'END{print $0}' | awk '$1 ~ /[0-9]+/{print $12}' | sed -E 's/[0-9]+/n/g' | sort | uniq -c

image_2023-03-04_20230304141731
可以看到,20个线程的线程池中,在1秒内只有4个线程是活跃的,线程池中线程数量是足够的。

这个命令脚本就不展开解释了,也不复杂,有linux命令基础的将命令依次拆开执行,应该能Get到脚本逻辑,没学过linux命令的话,就直接拿去用吧😅

查看活跃连接数

在Linux上,使用ss -natp|grep pid=1可以查看1号进程的TCP连接,如下:
image_2023-03-04_20230304144050
比如若redis数据库端口是6379的话,那么可这样查看redis连接池中连接数量,如下:

$ ss -natp | grep pid=1 | awk '$5~/:6379$/' | wc -l
20

可见当前有20个redis网络连接,那同样的,其中有多少个是活跃的呢?

经过查看man文档,发现ss中也有一个-i选项,如下:
image_2023-03-04_20230304145652
可以发现,添加-i选项后,ss会输出tcp连接中的一些额外信息,其中lastsnd表示最后一次发送包到当前所经历的毫秒数,lastrcv表示最后一次接收包到当前所经历的毫秒数。

有了这个信息后,就可以通过awk过滤出lastsnd或lastrcv小于1000的tcp连接,这些连接即是1秒内活跃过的连接了,因此我又编写了如下命令脚本。

$ ss -natpi | sed '1!{N;s/\n//;}' | grep pid=1 | awk -v t=1000 'match($0,/lastsnd:(\w+) lastrcv:(\w+)/,a) && (a[1]<t || a[2]<t) && match($4,/(.+):(\w+)$/,s) && match($5,/(.+):(\w+)$/,d) && s[2]>=32768{print d[2]}' |sort |uniq -c |sort -nk2
      8 80
      3 3306
      7 3307
      6 6379
      1 7916

如上,可以看到各连出端口的活跃连接情况,其中80是http连接池端口,3306与3307是MySQL主从库的连接池端口,6379是redis连接池的端口。

这是java应用主动连出连接的活跃情况,那调用方连入java应用的呢?

其实只需要稍微调整一下awk脚本即可,如下:

  1. s[2]>=32768调整为s[2]<32768,其中32768是Linux默认的临时端口号的分界线,可通过sysctl net.ipv4.ip_local_port_range查询,本地端口号大于这个值,代表是连出连接.
  2. print d[2]调整为print s[2],和上面条件联合起来,输出的就是本地监听端口了.

调整后,效果如下:

$ ss -natpi | sed '1!{N;s/\n//;}' | grep pid=1 | awk -v t=1000 'match($0,/lastsnd:(\w+) lastrcv:(\w+)/,a) && (a[1]<t || a[2]<t) && match($4,/(.+):(\w+)$/,s) && match($5,/(.+):(\w+)$/,d) && s[2]<32768{print s[2]}' |sort |uniq -c |sort -nk2
     8 8080

可以发现,我们服务的8080端口,1秒内活跃过的连接数是8个。

注:只有当调用方也使用连接池时,这种方法获取到的活跃连接数才是准确的,若调用方使用短链接的话,则不准确。

arthas查看活跃线程数与连接数

通过上面的方法,已经可以查看活跃线程数与连接数了,但有些情况下,会丧失一些细节,如下:

  1. top中的线程名会截断,如果不同线程池的线程名前16字符一样,则在top中无法区分。
  2. ss中是通过端口来区分线程池的,但http服务的端口号基本都是80或443,所以不同域名的http服务的连接池无法区分。

若需要分辩这些细节,还是要深入到jvm里面来,而arthas就是一个不错的工具,它的vmtool命令能够获取指定类型的Java对象,并从Java对象中获取信息。

以springboot为例,获取内置tomcat线程池的活跃情况,如下:

# --action getInstances:表示获取对象实例
# --classLoaderClass:指定类加载器
# --className:指定要获取哪个类的实例
# --express:指定ognl表达式,用来从对象上获取信息
[arthas@1]$ vmtool --action getInstances --classLoaderClass org.springframework.boot.loader.LaunchedURLClassLoader --className org.apache.tomcat.util.threads.ThreadPoolExecutor  --express 'instances.{ #{"ActiveCount":getActiveCount(),"LargestPoolSize":getLargestPoolSize(),"CorePoolSize":getCorePoolSize(),"MaximumPoolSize":getMaximumPoolSize(),"QueueSize":getQueue().size(),"ThreadName":getThreadFactory().namePrefix }}' -x 2

image_2023-03-04_20230304155857
上面其实就是通过vmtool工具,获取到了tomcat的线程池对象,然后调用线程池的getActiveCount()等方法,获取到了活跃线程数🙄

要获取连接池的活跃情况,也一并呈上吧,如下:

# 获取druid连接池的使用情况
[arthas@1]$ vmtool --action getInstances --classLoaderClass org.springframework.boot.loader.LaunchedURLClassLoader --className com.alibaba.druid.pool.DruidDataSource  --express 'instances.{ #{"url":#this.getUrl().split("\\?")[0], "username":#this.getUsername(),"PoolingCount":#this.getPoolingCount(),"ActiveCount":#this.getActiveCount(),"MaxActive":#this.getMaxActive(),"WaitThreadCount":#this.getWaitThreadCount(),"MaxWaitThreadCount":#this.getMaxWaitThreadCount()} }' -x 2

# 获取httpclient连接池的使用情况
[arthas@1]$ vmtool --action getInstances --classLoaderClass org.springframework.boot.loader.LaunchedURLClassLoader --className org.apache.http.impl.conn.PoolingHttpClientConnectionManager --express 'instances.{ #pool=#this.pool.routeToPool.values() }' -x2

可以看到,arthas真的很方便实用,对于Java Boy来说,值得好好研究研究👍👍👍

  • 1
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
### 回答1: 要查看Linux系统中的CPU线程数,可以使用以下命令: 1. 使用lscpu命令查看CPU信息,包括CPU核心数、线程数等信息。 2. 使用cat /proc/cpuinfo命令查看CPU信息,其中processor字段表示CPU核心数,而siblings字段表示CPU线程数。 3. 使用nproc命令查看CPU核心数。 4. 使用top或htop命令查看CPU使用情况,其中的CPU行显示了CPU核心数和线程数。 希望能对您有所帮助! ### 回答2: 在Linux系统中,可以使用不同的命令来查看CPU线程数。 1. top命令: top命令是一种实时的系统监视器,它可以实时监控系统的CPU使用率、进程、内存、负载平均值等。在终端输入top命令后,可以看到系统的实时运行情况,其中第二行的“Tasks”项显示了系统的进程总数和线程总数。 2. lscpu命令: lscpu是一个Linux系统命令,它用于显示CPU架构信息,包括CPU型号、核数、线程数和缓存大小等。在终端输入lscpu命令后,可以看到CPU架构的详细信息,其中“Thread(s) per core”项显示了每个CPU核心的线程数。 3. /proc/cpuinfo文件: /proc/cpuinfo是一个虚拟的文件系统,在Linux系统中它存储了系统的CPU信息。可以使用命令“cat /proc/cpuinfo”打印出该文件内容,其中每个处理器都有一个独立的区域,其中Thread(s)项显示了该处理器的线程数。 总结:Linux查看CPU线程数的方法有很多,包括使用top命令、lscpu命令和/proc/cpuinfo文件等。以上三种方法都是比较简单的,您可以根据自己的需要选择一种适合自己的方式来查看CPU线程数。 ### 回答3: 在Linux系统中,可以使用多种命令来查看CPU线程数,其中就包括了top命令、htop命令、lscpu命令、nproc命令、以及/proc/cpuinfo文件等。下面我们就来逐一介绍这些命令的使用方法。 1. top命令 top命令是一个非常常用的Linux系统命令,它可以用于查看系统中CPU、内存等各种资源的使用情况。在top命令的输出结果中,包含了每个CPU核心的线程数。输入top命令后,可以按下1键,就可以看到每个CPU核心的线程数了。 2. htop命令 htop命令与top命令十分类似,也是用于查看系统资源的使用情况。不同的是,htop命令的输出结果更加详细、易读,同时还支持使用鼠标进行交互式操作。在htop命令的输出结果中,也可以很容易地看到每个CPU核心的线程数。 3. lscpu命令 lscpu命令可以用于查看CPU的相关信息,包括CPU架构、CPU核心数、线程数等等。输入lscpu命令后,可以看到类似于下面的输出结果: CPU(s): 4 Thread(s) per core: 2 Core(s) per socket: 2 Socket(s): 1 4. nproc命令 nproc命令可以用于查看当前系统中CPU核心数和线程数。输入nproc命令后,可以直接看到当前系统中的线程数。 5. /proc/cpuinfo文件 最后一个方法则是直接查看/proc/cpuinfo文件。这个文件包含了当前系统中CPU的详细信息,包括CPU型号、频率、缓存大小、核心数量、线程数等等。在这个文件中,每个CPU核心对应着一个独立的section。因此,我们可以非常简单地通过查看该文件来获取CPU线程数的信息。 总之,以上这些方法都可以用来查看Linux系统中的CPU线程数。不同的方法有各自的特点,可以根据实际情况选择合适的方法来使用。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值