- ubuntu12.04环境下使用kvm ioctl接口实现最简单的虚拟机
- Ubuntu 通过无线网络安装Ubuntu Server启动系统后连接无线网络的方法
- 在Ubuntu上搭建网桥的方法
- ubuntu 虚拟机上网方式及相关配置详解
CFSDN坚持开源创造价值,我们致力于搭建一个资源共享平台,让每一个IT人在这里找到属于你的精彩世界.
这篇CFSDN的博客文章JVM FULL GC 生产问题 II-如何定位内存泄露?由作者收集整理,如果你对这篇文章有兴趣,记得点赞哟.
情景回顾 。
我们在上一篇 JVM FULL GC 生产问题笔记 中提出了如何更好的实现一个多线程消费的实现方式.
没有看过的小伙伴建议看一下.
本来以为一切都可以结束的,不过又发生了一点点意外,这里记录一下,避免自己和小伙伴们踩坑.
。
简介 。
上一节中我们尝试了多种多线程方案,总会有各种各样奇怪的问题.
于是最后决定使用生产-消费者模式去实现.
实现如下:
这里使用 AtomicLong 做了一个简单的计数.
userMapper.handle2(Arrays.asList(user)); 这个方法是同事以前的方法,当然做了很多简化.
就没有修改,入参是一个列表。这里为了兼容,使用 Arrays.asList() 简单封装了一下.
测试验证 。
当然这个方法在集成环境跑没有任何的问题.
于是就开始直接上生产验证,结果开始很快,然后就可以变慢了.
一看 GC 日志,梅开二度,FULL GC.
可恶,圣斗士竟然会被同一招打败 2 次吗?
FULL GC 的产生 。
一般要发现 full gc,最直观的感受就是程序很慢.
这时候你就需要添加一下 GC 日志打印,看一下是否有 full gc 即可.
这个最坑的地方就在于,性能问题是测试一般无法验证的,除非你进行压测.
压测还要同时满足两个条件:
(1)数据量足够大,或者说 QPS 足够高。持续压 。
(2)资源足够少,也就是还想马儿跑,还想马儿不吃草.
好巧不巧,我们同时赶上了两点.
那么问题又来了,如何定位为什么 FULL GC 呢?
内存泄露 。
程序变慢并不是一开始就慢,而是开始很快,然后变慢,接着就是不停的 FULL GC.
这就和自然的想到是内存泄露.
如何定位内存泄露呢?
你可以分成下面几步:
(1)看代码,是否有明显存在内存泄露的地方。然后修改验证。如果无法解决,则找出可能存在问题的地方,执行第二步.
(2)把 FULL GC 时的堆栈信息 dump 下来,分析到底是什么数据过大,然后结合 1 去解决.
接下来,让我们一起看一下这个过程的简化版本记录.
。
看代码 。
最基本的生产者-消费者模式确认了即便,感觉没啥问题.
于是就要看一下消费者模式中调用其他人的方法问题.
方法的核心目的 。
(1)遍历入参列表,执行业务处理.
(2)把当前批次的处理结果写入到文件中.
方法实现 。
简化版本如下:
这种代码怎么说呢,大概就是祖传代码吧,不晓得大家有没有见过,或者写过呢?
我们可以不看文件部分,核心部分实际上只有:
代码存在的问题 。
你觉得上面的代码有哪些问题?
什么地方可能存在内存泄露呢?
有应该如何改进呢?
看堆栈 。
如果你看代码已经确定了疑惑的地方,那么接下来就是去看一下堆栈,验证下自己的猜想.
堆栈的查看方式 。
jvm 堆栈查看的方式很多,我们这里以 jmap 命令为例.
(1)找到 java 进程的 pid 。
你可以执行 jps 或者 ps ux 等,选择一个你喜欢的.
我们 windows 本地测试了下(实际生产一般是 linux 系统):
UserServicePageQueue 是我们执行的测试程序,所以 pid 是 11964 。
(2)执行 jmap 获取堆栈信息 。
命令:
效果如下:
当然下面还有很多,你可以使用 head 命令过滤.
当然,如果服务器不支持这个命令,你可以把堆栈信息输出到文件中:
堆栈分析 。
我们可以很明显发现不合理的地方:
[C 这里指的是 chars,有 161031.
String 是字符串,有 157949.
当然还有 User 对象,有 139358.
我们每一次分页是 1W 个,queue 中最多是 19999 个,这么多对象显然不合理.
。
chars 和 String 为什么这么多 。
代码给人的第一感受,就是和业务逻辑没啥关系的写文件了.
很多小伙伴肯定想到了可以使用 TWR 简化一下代码,不过这里存在两个问题:
(1)最后文件中能记录所有的执行结果吗?
(2)有没有更好的方式呢?
对于问题1,答案是不能。虽然我们为每一个线程创建一个文件,但是实际测试,发现文件会被覆盖.
实际上比起我们自己写文件,更应该使用 log 去记录结果,这样更加优雅.
于是,最后把代码简化如下:
user 对象为什么这里多?
我们看一下核心业务代码:
这里在判断是否存在的时候构建了一个 mybatis 中常用的 User 查询条件,然后判断查询的列表大小.
这里有两个问题:
(1)判断是否存在,最好使用 count,而不是判断列表结果大小.
(2)User userExample 的作用域尽量小一点.
调整如下:
调整之后的代码 。
这里的 System.out.println 实际使用时用 log 替代,这里只是为了演示.
生产验证 。
全部改完之后,重新部署验证,一切顺利.
希望不会有第三篇。:) 。
小结 。
当然验证的过程中还发生过一点小插曲,比如开发没有权限看堆栈信息,执行命令时程序已经假死等等.
生产 full gc 是一个比较麻烦的问题,一个是难以复现,另一个是如果是偶发性的,又是实时链路,可能也不好执行 dump 命令.
所以写代码还是写的尽可能简单的好,不然会有各种问题.
能复用已有的工具、中间件尽量复用.
这样看来,我们自己写的生产-消费者模式也不太好,因为复用性不强,所以建议使用公司已有的 mq 工具,不过如何选择,还是看具体的业务场景.
架构,就是权衡.
希望本文对你有所帮助.
【原文地址】:https://www.toutiao.com/i6949465946352255525/ 。
最后此篇关于JVM FULL GC 生产问题 II-如何定位内存泄露?的文章就讲到这里了,如果你想了解更多关于JVM FULL GC 生产问题 II-如何定位内存泄露?的内容请搜索CFSDN的文章或继续浏览相关文章,希望大家以后支持我的博客! 。
是否可以复制一个完整的 JVM,并且在故障转移的情况下只需将负载转移到复制的 JVM 上? 如果是,那我们该怎么做呢? 最佳答案 如果您的应用程序是 Web 应用程序,请阅读“集群”和“负载平衡”。大
我读了下面的话,但我想知道它们之间的区别...... JVM 规范、JVM 实现、JVM 运行时 最佳答案 JVM 规范:描述 JVM 应如何运行的文档。 JVM 实现:基于 JVM 规范的 JVM
我目前有四个不同的 java 应用程序,它们由 .bat 文件启动的 jar 运行,这些文件位于 Windows XP Embedded 开始菜单的 starup 文件夹中。我还启动了 Firefox
有人能给我一些关于强制 64 位 jvm 作为 32 位 jvm 运行的想法吗? 我需要为蓝牙连接编写一个 jse 桌面应用程序。为此,我需要实现 Bluecove jar 。它只有 32 位文件。所
我看到过关于这个问题的多条评论——有人说是,有人说不是,许多答案模棱两可。任何人都可以用更简单的术语描述它所在的位置吗?在一篇文章中,我什至看到有人说它与类加载器加载类的类内存共享相同的内存位置 -
我正在寻找所有可能的 jvm 退出代码的完整列表(不是 java System.exit(x))。我使用搜索引擎唯一能找到的是 SIGTERM 退出代码列表:http://journal.thobe.
为了监视任何正常的 Java 进程 JVM,我们可以使用 Attach API。是否有可用于监控 WebSphere JVM 的 API? 最佳答案 您可以使用 PMI(性能监控基础设施)来监控 JV
这个问题在这里已经有了答案: 8年前关闭。 Possible Duplicate: Java - C-Like Fork? 我想知道如何从 JDK fork 子 JVM,甚至有可能这样做吗? 一些框架
JVM 上的哪些图灵完备语言实现不使用 JVM 堆栈作为调用堆栈? (我问是因为我想在同一个线程中实现 Scala 和另一种语言之间的协程。) 最佳答案 闪蝶 SISC(方案代码的第二解释者) 曾经不
我看到here除了 Java 之外,还有很多语言可以在 JVM 上运行。我对在 JVM 中运行的其他语言的整个概念有些困惑。所以: 为 JVM 使用其他语言有什么优势? 为 JVM 编写语言/编译器需
我已经运行了 straced JVM (OpendJDK 11): strace -e trace=mmap java -Xms8192m Main 输出是: mmap(NULL, 8192, PRO
我已经运行了 straced JVM (OpendJDK 11): strace -e trace=mmap java -Xms8192m Main 输出是: mmap(NULL, 8192, PRO
我编写了一个简单的数独求解器。为了粗略测试性能,我使用简单的 System.currentTimeMillis 调用。 我在文本文件中准备了一组初始数独配置。该程序读取该文件并解决每个数独配置。运行测
JVM 被广泛使用:Scala、Groovy、Jython 等。我听说它被描述为“卓越”、“出色”和“严重低估”。为什么? 更具体地说,是什么让 JVM 独一无二?随着所有资金投入 .NET,或者 C
这个问题在这里已经有了答案: 10年前关闭。 Possible Duplicate: Are there any Java VMs which can save their state to a fi
想象一下 6-7 台服务器的设置都完全相同Java 版本“1.6.0_18”OpenJDK 运行时环境 (IcedTea6 1.8) (fedora-36.b18.fc11-i386)OpenJDK
(如有错误请指正) 我了解到,当您通过发出 java 命令来运行 java 程序时, java MyProg 程序将在新的 JVM 上运行。 什么将程序加载到新的 JVM 中?是生成新线程的 JRE
我们有一个使用 JNI 的桌面应用程序偶尔会导致 JVM 崩溃。幸运的是,JVM 会生成一个 hs_err_pidXXXX.log 文件,这对于调试此类错误非常有用。然而,它似乎总是转到当前工作目录,
我在命令提示符下运行一个程序集 jar 文件并得到下面的异常。并导致终止。 Uncaught error from thread [ccp-akka.persistence.dispatchers.d
一、什么是Java虚拟机 虚拟机:指以软件的方式模拟具有完整硬件系统功能、运行在一个完全隔离环境中的完整计算机系统 ,是物理机的软件实现。常用的虚拟机有VMWare,Visual Box,Java
我是一名优秀的程序员,十分优秀!