[java] JVM 内存泄漏分析案例

JVM 内存泄漏分析

初始代码

java 复制代码
public class App {

    private static final Map<Long, Object> productCache = new HashMap<>();

    private static final Random RANDOM = new Random();

    public static void main(String[] args) {
        try {
            while (true) {
                long id = System.nanoTime(); 
                productCache.put(id, new Object());
                
                if (productCache.size() % 1_000_000 == 0) {
                    System.out.println("The current number of cached objects: " + productCache.size());
                }
            }
        } catch (OutOfMemoryError e) {
            System.err.println("\n OOM!!! The final number of cached objects: " + productCache.size());
            e.printStackTrace();
        }
    }
}

编译运行

命令行在APP.java所在目录执行

复制代码
# 编译
javac App.java

# 运行,最大堆内存大小为 256MB,自动生成快照
java -Xms128m -Xmx256m -XX:+HeapDumpOnOutOfMemoryError -XX:HeapDumpPath=D:/tmp/demo1-heap.hprof App

分析内存快照

使用MAT1.7.0版本,MAT入门参考


修复代码

改用带淘汰机制的缓存替换 HashMapLinkedHashMap 并实现 LRU(最近最少使用)淘汰策略,限制缓存最大容量:

java 复制代码
public class App {

    // private static final Map<Long, Object> productCache = new HashMap<>();

    private static final Random RANDOM = new Random();

    private static final Map<Long, Object> productCache = new LinkedHashMap<Long, Object>(1024, 0.75f, true) {
        @Override
        protected boolean removeEldestEntry(Map.Entry<Long, Object> eldest) {
            // More than one million goods will be eliminated if they have not been used for the longest time
            return size() > 1_000_000;
        }
    };

    public static void main(String[] args) {
        try {
            while (true) {
                long id = System.nanoTime(); 
                productCache.put(id, new Object());
                
                if (productCache.size() % 1_000_000 == 0) {
                    System.out.println("The current number of cached objects: " + productCache.size());
                }
            }
        } catch (OutOfMemoryError e) {
            System.err.println("\n OOM!!! The final number of cached objects: " + productCache.size());
            e.printStackTrace();
        }
    }
}

再次编译运行发现缓存维持在1_000_000

常见内存泄漏场景及解决

根据分析结果,针对性修复:

静态集合未清理

问题:static List/Map 等容器长期持有对象引用,未及时移除(如缓存未设置过期策略)。

解决:改用弱引用集合(WeakHashMap)、设置缓存淘汰机制(如 LRU),或定期清理无效数据。

资源未关闭

问题:数据库连接、文件流、网络连接等资源未关闭,导致对象无法回收。

解决:使用 try-with-resources 自动关闭资源,或在 finally 块中显式释放。

监听器 / 回调未移除

问题:注册的监听器(如 GUI 事件、观察者模式)未注销,导致被监听对象长期引用。

解决:在对象销毁前移除监听器,或使用弱引用实现回调。

线程泄漏

问题:线程池核心线程持有大对象引用,或未正确停止的线程(如 Thread 未中断)。

解决:控制线程池核心线程数,使用 ThreadLocal 时注意清理(remove()),确保线程能正常终止。

类加载器泄漏

问题:自定义类加载器加载的类未被回收(如热部署场景),导致元空间溢出。

解决:避免长期持有类加载器引用,确保加载的类及其依赖可被 GC 回收。

相关推荐
期待のcode14 小时前
MyBatisX插件
java·数据库·后端·mybatis·springboot
醇氧16 小时前
【Windows】优雅启动:解析一个 Java 服务的后台启动脚本
java·开发语言·windows
sunxunyong16 小时前
doris运维命令
java·运维·数据库
菜鸟起航ing16 小时前
Spring AI 全方位指南:从基础入门到高级实战
java·人工智能·spring
古城小栈16 小时前
Docker 多阶段构建:Go_Java 镜像瘦身运动
java·docker·golang
MapGIS技术支持17 小时前
MapGIS Objects Java计算一个三维点到平面的距离
java·开发语言·平面·制图·mapgis
Coder_Boy_17 小时前
业务导向型技术日志首日记录(业务中使用的技术栈)
java·驱动开发·微服务
程序员zgh17 小时前
C++ 互斥锁、读写锁、原子操作、条件变量
c语言·开发语言·jvm·c++
小灰灰搞电子17 小时前
Qt 重写QRadioButton实现动态radioButton源码分享
开发语言·qt·命令模式
by__csdn18 小时前
Vue3 setup()函数终极攻略:从入门到精通
开发语言·前端·javascript·vue.js·性能优化·typescript·ecmascript