Java 解决堆内存溢出 | Eddie'Blog
Java 解决堆内存溢出

Java 解决堆内存溢出

eddie 296 2021-05-06

目录

实战-内存溢出

  • 堆内存溢出
  • 栈内存溢出
  • 方法区溢出
  • 直接内存溢出

实战-堆内存溢出

  • 演示堆内存溢出代码,并且定位问题
  • 总结堆内存溢出的场景与解决方案
  • 分析商城项目中可能存在堆内存溢出的代码并且解决

堆内存溢出演示代码

public class HeapOOMTest {
    private List<String> oomList = new ArrayList<>();

    public static void main(String[] args) {
        HeapOOMTest oomTest = new HeapOOMTest();
        while (true) {
            oomTest.oomList.add(UUID.randomUUID().toString());
        }
    }
}

VM Args:-Xms20m -Xmx20m -XX:+HeapDumpOnOutOfMemoryError

运行该代码的Main方法

java.lang.OutOfMemoryError: Java heap space
Dumping heap to java_pid18940.hprof ...
Heap dump file created [23854536 bytes in 0.061 secs]
Exception in thread "main" java.lang.OutOfMemoryError: Java heap space
	at java.base/java.lang.Long.fastUUID(Long.java:441)
	at java.base/java.lang.System$2.fastUUID(System.java:2136)
	at java.base/java.util.UUID.toString(UUID.java:395)
	at com.imooc.jvminaction.HeapOOMTest.main(HeapOOMTest.java:16)

在项目的根目录会自动创建 java_pid18940.hprof 的文件

Mac系统:mat.app

Win系统:Eclipse Memory Analyzer

Eclipse Memory Analyzer 分析流程
  • File -> Open Heap Dump,加载刚才产生的hprof文件,选择Leak Suspects Report
  • 在Overview, 鼠标移动到拼图,可以看出
    • java.lang.Thread @ 0xff1dbc28 main
      Shallow Size: 120 B RetainedSize: 10.9 MB (明显不正常的)
  • 点击 Leak Suspects: includes leak suspects and a system overview.
    • Problem Suspect 1 会提示方法占用空间的百分比
      • 点击 Details » Accumulated Objects in Dominator Tree
      • 点击 java.lang.Object[160065] @ 0xffe00000
        • List objects --> with incoming references (别人调用自己)

图片.png

Leak Suspects » Leaks » Problem Suspect 1 » Description » See stacktrace 也是可以看出问题的

图片.png

jvisualvm.exe 分析堆内存溢出

图片.png

图片.png

图片.png

堆内存溢出的场景

  • 内存泄露
    • 借助工具定位问题
  • 非内存泄露
    • -Xms -Xmx 分配不合理

商城项目可能存在堆内存溢出的方法

1.通过控制pageSize判断

if (pageSize > 100) {
    pageSize = 100;
}

2.SpringMVC方式

类头加注解

@Validated

pageSize的入参加注解

@Max(100)

图片.png


# Java