Java 线程究竟占用多少内存

2019 年 6 月 27 日 ImportNew

(给ImportNew加星标,提高Java技能)

编译:ImportNew/唐尤华

dzone.com/articles/how-much-memory-does-a-java-thread-take


更新:


JVM 并没有主动按照线程数量分配 “threads ’ 1MB”,这个错误来自 NMT 报告。Java 8 “committed memory”会自动置为 “reserved memory”。参见 <https://bugs.openjdk.java.net/browse/JDK-8191369>


”committed memory“大小由堆栈深度决定,感谢 [Thomas Stuefe][1] 的提醒。


---


[1]:https://dzone.com/users/3720329/tstuefe.html


在 Java 应用总内存消耗中,线程使用的内存占很大一部分。根据应用具体特点,比如 CPU 受限或者 IO 受限,可以选择对应技术限制新建的线程数量。对于 IO 受限应用可以创建线程池,由线程处理 IO 操作,比如在阻塞或等待状态从数据库读取数据、发送 HTTP 请求等。如果是计算密集型应用,可以使用像 Netty 这样的 HTTP 服务器,线程数量少、节省内存。接下来通过具体示例展示一个新线程需要消耗多少内存。


线程内存包括栈帧、局部变量、方法参数等,线程大小默认如下(KB):


```shell
$ java -XX:+PrintFlagsFinal -version | grep ThreadStackSize
intx CompilerThreadStackSize = 1024 {pd product} {default}
intx ThreadStackSize = 1024 {pd product} {default}
intx VMThreadStackSize = 1024 {pd product} {default}
```


1. Java 8 线程内存消耗


```shell
$ java -XX:+UnlockDiagnosticVMOptions -XX:NativeMemoryTracking=summary /
-XX:+PrintNMTStatistics -version
openjdk version "1.8.0_212"
OpenJDK Runtime Environment (AdoptOpenJDK)(build 1.8.0_212-b03)
OpenJDK 64-Bit Server VM (AdoptOpenJDK)(build 25.212-b03, mixed mode)
Native Memory Tracking:
Total: reserved=6621614KB, committed=545166KB
- Java Heap (reserved=5079040KB, committed=317440KB)
(mmap: reserved=5079040KB, committed=317440KB)
- Class (reserved=1066074KB, committed=13786KB)
(classes #345)
(malloc=9306KB #126)
(mmap: reserved=1056768KB, committed=4480KB)
- Thread (reserved=19553KB, committed=19553KB)
(thread #19)
(stack: reserved=19472KB, committed=19472KB)
(malloc=59KB #105)
(arena=22KB #34)
```


可以看到两种类型的内存:


  • “Reserved:” 由操作系统承诺的可用内存大小。但尚未分配,JVM 无法访问

  • ”Committed:“ 已被 JVM 分配,可访问


在“Thread”部分可以看到,‘reserved’ 和 ‘committed’ 大小相同,接近于“线程数 ‘ 1MB”。这时因为 JVM 一开始就尽可能地为线程分配内存。


2. Java 11 线程内存消耗


```shell
$ java -XX:+UnlockDiagnosticVMOptions -XX:NativeMemoryTracking=summary /
-XX:+PrintNMTStatistics -version
openjdk version "11.0.2" 2019-01-15
OpenJDK Runtime Environment AdoptOpenJDK (build 11.0.2+9)
OpenJDK 64-Bit Server VM AdoptOpenJDK (build 11.0.2+9, mixed mode)
Native Memory Tracking:
Total: reserved=6643041KB, committed=397465KB
- Java Heap (reserved=5079040KB, committed=317440KB)
(mmap: reserved=5079040KB, committed=317440KB)
- Class (reserved=1056864KB, committed=4576KB)
(classes #426)
( instance classes #364, array classes #62)
(malloc=96KB #455)
(mmap: reserved=1056768KB, committed=4480KB)
( Metadata: )
( reserved=8192KB, committed=4096KB)
( used=2849KB)
( free=1247KB)
( waste=0KB =0,00%)
( Class space:)
( reserved=1048576KB, committed=384KB)
( used=270KB)
( free=114KB)
( waste=0KB =0,00%)
- Thread (reserved=15461KB, committed=613KB)
(thread #15)
(stack: reserved=15392KB, committed=544KB)
(malloc=52KB #84)
(arena=18KB #28)
```


从上面地结果可以注意到,Java 11 节省了很多内存,不会在线程创建时主动分配 ‘Reserved Memory’。当然,这只是 ’java -version‘ 命令的执行结果。如果试着使用 Java 11,会发现内存分配有显著改进。


感谢阅读,欢迎在下面评论。


推荐阅读

(点击标题可跳转阅读)

Java 多线程:volatile 变量、happens-before 关系及内存一致性

Java 内存模型

内存不足:杀死进程还是牺牲子进程


看完本文有收获?请转发分享给更多人

关注「ImportNew」,提升Java技能

好文章,我在看❤️

登录查看更多
6

相关内容

Java 是一门编程语言,拥有跨平台、面向对象、泛型编程等特性。
Python分布式计算,171页pdf,Distributed Computing with Python
专知会员服务
107+阅读 · 2020年5月3日
专知会员服务
44+阅读 · 2020年3月6日
专知会员服务
26+阅读 · 2020年2月15日
模型压缩究竟在做什么?我们真的需要模型压缩么?
专知会员服务
27+阅读 · 2020年1月16日
【干货】大数据入门指南:Hadoop、Hive、Spark、 Storm等
专知会员服务
95+阅读 · 2019年12月4日
在K8S上运行Kafka合适吗?会遇到哪些陷阱?
DBAplus社群
9+阅读 · 2019年9月4日
GPU 显存不足怎么办?
AINLP
13+阅读 · 2019年8月16日
已删除
架构文摘
3+阅读 · 2019年4月17日
R_leaflet包_最易上手地图教程(一)
R语言中文社区
10+阅读 · 2019年3月6日
Python高级技巧:用一行代码减少一半内存占用
AI研习社
5+阅读 · 2018年11月28日
实战 | 用Python做图像处理(三)
七月在线实验室
15+阅读 · 2018年5月29日
Python 杠上 Java、C/C++,赢面有几成?
CSDN
6+阅读 · 2018年4月12日
Spark的误解-不仅Spark是内存计算,Hadoop也是内存计算
Arxiv
5+阅读 · 2018年5月1日
Arxiv
14+阅读 · 2018年4月18日
Arxiv
7+阅读 · 2018年3月22日
VIP会员
相关VIP内容
相关资讯
在K8S上运行Kafka合适吗?会遇到哪些陷阱?
DBAplus社群
9+阅读 · 2019年9月4日
GPU 显存不足怎么办?
AINLP
13+阅读 · 2019年8月16日
已删除
架构文摘
3+阅读 · 2019年4月17日
R_leaflet包_最易上手地图教程(一)
R语言中文社区
10+阅读 · 2019年3月6日
Python高级技巧:用一行代码减少一半内存占用
AI研习社
5+阅读 · 2018年11月28日
实战 | 用Python做图像处理(三)
七月在线实验室
15+阅读 · 2018年5月29日
Python 杠上 Java、C/C++,赢面有几成?
CSDN
6+阅读 · 2018年4月12日
Spark的误解-不仅Spark是内存计算,Hadoop也是内存计算
相关论文
Top
微信扫码咨询专知VIP会员