什么是内存泄漏?

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回收
    }
}
相关推荐
程序员小假2 小时前
我们来说说当一个线程两次调用 start() 方法会出现什么情况?
java·后端
SimonKing2 小时前
Archery:开源、一站式的数据库 SQL 审核与运维平台
java·后端·程序员
皮皮林55114 小时前
IDEA 源码阅读利器,你居然还不会?
java·intellij idea
卡尔特斯18 小时前
Android Kotlin 项目代理配置【详细步骤(可选)】
android·java·kotlin
白鲸开源18 小时前
Ubuntu 22 下 DolphinScheduler 3.x 伪集群部署实录
java·ubuntu·开源
ytadpole18 小时前
Java 25 新特性 更简洁、更高效、更现代
java·后端
纪莫18 小时前
A公司一面:类加载的过程是怎么样的? 双亲委派的优点和缺点? 产生fullGC的情况有哪些? spring的动态代理有哪些?区别是什么? 如何排查CPU使用率过高?
java·java面试⑧股
JavaGuide19 小时前
JDK 25(长期支持版) 发布,新特性解读!
java·后端
用户37215742613519 小时前
Java 轻松批量替换 Word 文档文字内容
java