什么是内存泄漏?

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回收
    }
}
相关推荐
karry_k5 分钟前
PostgreSQL 在 MyBatis 中执行正常 SQL 失效:一次 DELETE USING 踩坑记录
java·后端
SamDeepThinking4 小时前
从源码到代码:MyBatis-Flex 与 MyBatis-Plus 的逐项对比
java·后端·程序员
她的男孩7 小时前
Spring Boot 接 Flowable 工作流:用 3 个注解搭一个请假审批流程
java·后端·架构
荣码8 小时前
LLM结构化输出:让AI返回JSON而不是废话,我踩了4个坑
java·python
plainGeekDev10 小时前
Gson → kotlinx.serialization
android·java·kotlin
小bo波18 小时前
Java Swing 图形用户界面实验 —— 从算术练习到游戏开发的完整实践
java·课程设计·gui·游戏开发·扫雷·swing
咖啡八杯20 小时前
GoF设计模式——备忘录模式
java·后端·spring·设计模式
SamDeepThinking1 天前
裁掉那个差程序员后,给你看团队里高手的代码:这个习惯,希望你有
java·后端·程序员