Java 深入理解内存泄漏

2019-01-01  本文已影响0人  Little丶Jerry

Java 的一个最显著的优势是内存管理。你只需要简单的创建对象而不需要负责释放空间,因为 Java 的垃圾回收器会负责内存的回收。然而,情况并不是这样简单,内存泄露还是经常会在 Java 应用程序中出现。

一、内存泄漏

内存泄露的定义:对于应用程序来说,当对象已经不再被使用,但是 Java 的垃圾回收器不能回收它们的时候,就产生了内存泄露。

要理解这个定义,我们需要理解对象在内存中的状态。如下图所示,展示了哪些对象是无用对象,哪些是未被引用的对象;

未引用对象将会被垃圾回收器回收,而引用对象却不会。未引用对象很显然是无用的对象。然而,无用的对象并不都是未引用对象,有一些无用对象也有可能是引用对象,这部分对象正是内存泄露的来源。

二、内存泄漏发生的原因

如下图所示,对象 A 引用对象 B,A 的生命周期(t1-t4)比 B 的生命周期(t2-t3)要长,当 B 在程序中不再被使用的时候,A 仍然引用着 B。在这种情况下,垃圾回收器是不会回收 B 对象的,这就可能造成了内存不足问题,因为 A 可能不止引用着 B 对象,还可能引用其它生命周期比 A 短的对象,这就造成了大量无用对象不能被回收,且占据了昂贵的内存资源。

同样的,B 对象也可能引用着一大堆对象,这些被B对象引用着的对象也不能被垃圾回收器回收,所有的这些无用对象消耗了大量内存资源。

三、造成内存泄露的常见情形

Static Vector v = new Vector(10);
for (int i = 1; i<100; i++) {
    Object o = new Object();
    v.add(o);
    o = null;
}
import java.util.HashSet;
import java.util.Set;

public class MemoryOut {
    public static void main(String[] args) {
        Set<Person> set = new HashSet<Person>();
        Person p1 = new Person("唐僧","pwd1",25);
        Person p2 = new Person("孙悟空","pwd2",26);
        Person p3 = new Person("猪八戒","pwd3",27);
        set.add(p1);
        set.add(p2);
        set.add(p3);
        System.out.println("总共有:"+set.size()+" 个元素!"); //结果:总共有:3 个元素!
        p3.setAge(2); //修改p3的年龄,此时p3元素对应的hashcode值发生改变
        set.remove(p3); //此时remove不掉,造成内存泄漏
        set.add(p3); //重新添加,居然添加成功
        System.out.println("总共有:"+set.size()+" 个元素!"); //结果:总共有:4 个元素!
        for (Person person : set) {
            System.out.println(person);
        }
    }
}

class Person {
    int age;
    String name;
    String password;

    public Person(String name, String password, int age) {
        this.name = name;
        this.password = password;
        this.age = age;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    @Override
    public int hashCode() {
        final int prime = 31;
        int result = 1;
        result = prime * result + age;
        return result;
    }

    @Override
    public boolean equals(Object obj) {
        if (this == obj)
            return true;
        if (obj == null)
            return false;
        if (getClass() != obj.getClass())
            return false;
        Person other = (Person) obj;
        if (age != other.age)
            return false;
        return true;
    }   
}
class A {
    public A() {
        B.getInstance().setA(this);
    }
    ....
}

//B类采用单例模式
class B {
    private A a;
    private static B instance=new B();

    public B(){}
    public static B getInstance() {
        return instance;
    }

    public void setA(A a) {
        this.a = a;
    }

    //getter...
}

显然 B 采用 singleton 模式,它持有一个 A 对象的引用,而这个 A 类的对象将不能被回收。想象下如果 A 是个比较复杂的对象或者集合类型会发生什么情况。

四、内存泄露的解决方案

作者:六尺帐篷
链接:https://www.jianshu.com/p/d2823693ccc2

上一篇 下一篇

猜你喜欢

热点阅读