什么是内存泄漏?

1.什么是内存泄漏?

​ 程序运行过程中会不断的分配内存空间,那些不再使用的内存空间应该即时被回收,从而保证系统可以再次使用这些内存,如果存在无用的内存没有被回收回来,那就是内存泄漏。

​ 对于我们java程序员来说,所有不可达的对象都由垃圾回收机制来负责回收,因此java程序员不需要考虑这部分的内存泄漏。但是如果程序中有一些java对象,它们处于可达状态,但是程序以后永远都不会再访问它们,那么它们所占用的内存空间也不会被回收,它们所占用的空间也会产生内存泄漏,对于java程序而言,只要它们一直处于可达状态,垃圾回收机制就不会回收它们,即使他们对于程序来说已经变成了垃圾(因为程序再也不需要它们了),而对于垃圾回收机制来说,它们还不是垃圾(因为对象还处于可达状态),因此不能进行回收。

​ 以下是一个存在内存泄漏问题的代码例子:

arduino 复制代码
public class StaticCollectionLeak {
	//static静态变量的生命周期:从类加载到JVM关闭,始终存在于内存中。
    private static final List<Object> LEAK_LIST = new ArrayList<>();
    
    public void addToList(byte[] obj){
        LEAK_LIST.add(obj);// obj对象被LEAK_LIST静态集合引用着,无法被回收
    }

    public static void main(String[] args) {
        StaticCollectionLeak leak = new StaticCollectionLeak();
        for (int i = 0; i < 1000000; i++){
            leak.addToList(new byte[1024*1024]);// 每次添加1MB对象
        }
        // LEAK_LIST未被清理,导致内存泄漏
    }
}

​ 解决方法1:在程序结束时清理静态集合(推荐简单场景)

typescript 复制代码
public class StaticCollectionLeak {
    private static final List<Object> LEAK_LIST = new ArrayList<>();

    public void addToList(Object obj) {
        LEAK_LIST.add(obj);
    }

    // 新增清理方法
    public static void clearList() {
        LEAK_LIST.clear();
    }

    public static void main(String[] args) {
        StaticCollectionLeak leak = new StaticCollectionLeak();
        try {
            for (int i = 0; i < 100; i++) {
                leak.addToList(new byte[1024 * 1024]);
            }
        } finally {
            // 确保在程序结束时清理
            clearList();
        }
    }
}

​ 解决方法2:改为非静态集合(推荐常规场景)

typescript 复制代码
public class SafeCollection {
    private final List<Object> tempList = new ArrayList<>();

    public void addToList(Object obj) {
        tempList.add(obj);
    }

    // 需要时手动清理
    public void clear() {
        tempList.clear();
    }

    public static void main(String[] args) {
        SafeCollection safe = new SafeCollection();
        for (int i = 0; i < 100; i++) {
            safe.addToList(new byte[1024 * 1024]);
            // 业务逻辑完成后清理
            if (i % 50 == 0) {
                safe.clear();
            }
        }
        // 对象销毁时自动GC回收
    }
}
相关推荐
可观测性用观测云20 小时前
Pyroscope Java 接入最佳实践
java
气π20 小时前
【JavaWeb】——(若依 + AI)-基础学习笔记
java·spring boot·笔记·学习·java-ee·mybatis·ruoyi
阿里云云原生21 小时前
AgentScope Java 1.0:从模型到应用,AI Agent 全生命周期管理利器!
java·云原生
running up21 小时前
Maven依赖管理和项目构建工具
java·maven
老华带你飞21 小时前
列车售票|基于springboot 列车售票系统(源码+数据库+文档)
java·数据库·vue.js·spring boot·后端·学习·spring
java_logo21 小时前
LinuxServer.io LibreOffice 容器化部署指南
java·开发语言·docker·dubbo·openoffice·libreoffice·opensource
qq_589568101 天前
@NotBlank与@NotEmpty注解无法导入
java·开发语言
李拾叁的摸鱼日常1 天前
Spring Boot中OncePerRequestFilter原理与Filter单次调用控制全解析
java·后端
script.boy1 天前
基于spring boot校园二手交易平台的设计与实现
java·spring boot·后端
爱潜水的小L1 天前
自学嵌入式day30,回收进程
java·linux·服务器