Java虚拟机进行生产环境线上问题解决以及性能问题的优化。
解决内存溢出的步骤总共分为四个步骤,其中前两个步骤是最核心的:
VisualVM是多功能合一的Java故障排除工具并且他是一款可视化工具,整合了 命令行 JDK 工具和轻量级分析功能,功能非常强大。这款软件在Oracle JDK 6~8 中发布,但是在 Oracle JDK 9 之后不在JDK安装目录下需要单独下载。下载地址:VisualVM: Home
Arthas 是一款线上监控诊断产品,通过全局视角实时查看应用 load、内存、gc、线程的状态信息,并能在不修改应用代码的情况下,对业务问题进行诊断, 包括查看方法调用的出入参、异常,监测方法执行耗时,类加载信息等,大大提升线上问题排查效率。arthas (aliyun.com)
优点:
Prometheus+Grafana是企业中运维常用的监控方案,其中Prometheus用来采集系统或者应用的相关数据,同时具备告警功能。Grafana可以将Prometheus采集到的数据以可视化的方式进行展示。
优点:
实际上,代码中的内存泄漏大部分能在测试环境上被排除。在测试环境上采用压力测试,发送大量的请求到服务端测试。
问题:在定义新类时没有重写正确的equals()和hashCode()方法。在使用HashMap的场景下,如果使用这个类对象作为key,HashMap在判断key是否已经存在时会使用这些方法,如果重写方式不正确,会导致相同的数据被保存多份。
问题:
解决方案:
问题:如果仅仅使用手动创建的线程,就算没有调用ThreadLocal的remove方法清理数据,也不会产生内存泄漏。因为当线程被回收时,ThreadLocal也同样被回收。但是如果使用线程池就不一定了。
解决方案:线程方法执行完,一定要调用ThreadLocal中的remove方法清理对象。
问题:如果大量的数据在静态变量中被长期引用,数据就不会被释放,如果这些数据不再使用,就成为了内存泄漏。
解决方案:
问题:连接和流这些资源会占用内存,如果使用完之后没有关闭,这部分内存不一定会出现内存泄漏
解决方案:从 Java 7 开始,使用try-with-resources语法可以用于自动关闭资源。
并发请求问题指的是用户通过发送请求向Java应用获取数据,正常情况下Java应用将数据返回之后,这部分数据就可以在内存中被释放掉。但是由于用户的并发请求量有可能很大,同时处理数据的时间很长,导致大量的数据存在于内存中,最终超过了内存的上限,导致内存溢出。
如果并未产生内存溢出,也可以导出运行中系统的内存快照,比较简单的方式有两种,注意只需要导出标记为存活的对象:
jmap -dump:live,format=b,file=文件路径和文件名 进程ID
heapdump --live 文件路径和文件名
我们用下面的程序来模拟内存溢出的场景:
/**
* 内存溢出的参数
* -Xms20m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=D:\Develop_Tools\Java\hprof
* 分析内存溢出:
* 1.占用内存过大的对象有哪些(Histogram)
* 2.被谁引用的(dominator_tree)
* 3.定位到具体的代码(thread_overview)
*/
public class OOMTest {
public static void main(String[] args) {
List<byte[]> memoryLeakArray=new ArrayList<>();
for(int i=0;i<100;i++){
byte[] temp=new byte[1024*1024];//添加 1m 的数据到list中
memoryLeakArray.add(temp);
}
}
}
这里我使用的内存分析工具是 Eclipse Memory Analyzer,分析内存溢出主要分3个步骤:
Histogram(直方图):查看内存占用
dominator_tree(支配树):关系引用图
thread_overview:查看线程的概览信息
定位到了具体的26行代码
在程序员开发用的机器内存范围之内的快照文件,直接使用MAT打开分析即可。但是经常会遇到服务器上的程序占用的内存达到10G以上,开发机无法正常打开此类内存快照,我们可以找一台内存足够的服务器,在服务器上使用MAT。
此时需要下载服务器操作系统对应的MAT。下载地址:Eclipse Memory Analyzer Open Source Project | The Eclipse Foundation
然后通过MAT中的脚本生成分析报告(体积很小,包括了一些静态页面),下载到开发机上
./ParseHeapDump.sh 快照文件路径 org.eclipse.mat.api:suspects
org.eclipse.mat.api:overview org.eclipse.mat.api:top_components
注意:默认MAT分析时只使用了1G的堆内存,如果快照文件超过1G,需要修改MAT目录下的 MemoryAnalyzer.ini配置文件调整最大堆内存。