Java 的一个最显著的优势是内存管理。你只需要简单的创建对象而不需要负责释放空间,因为 Java 的垃圾回收器会负责内存的回收。然而,情况并不是这样简单,内存泄露还是经常会在 Java 应用程序中出现。
一、内存泄漏
内存泄露的定义:对于应用程序来说,当对象已经不再被使用,但是 Java 的垃圾回收器不能回收它们的时候,就产生了内存泄露。
要理解这个定义,我们需要理解对象在内存中的状态。如下图所示,展示了哪些对象是无用对象,哪些是未被引用的对象;
未引用对象将会被垃圾回收器回收,而引用对象却不会。未引用对象很显然是无用的对象。然而,无用的对象并不都是未引用对象,有一些无用对象也有可能是引用对象,这部分对象正是内存泄露的来源。
二、内存泄漏发生的原因
如下图所示,对象 A 引用对象 B,A 的生命周期(t1-t4)比 B 的生命周期(t2-t3)要长,当 B 在程序中不再被使用的时候,A 仍然引用着 B。在这种情况下,垃圾回收器是不会回收 B 对象的,这就可能造成了内存不足问题,因为 A 可能不止引用着 B 对象,还可能引用其它生命周期比 A 短的对象,这就造成了大量无用对象不能被回收,且占据了昂贵的内存资源。
同样的,B 对象也可能引用着一大堆对象,这些被B对象引用着的对象也不能被垃圾回收器回收,所有的这些无用对象消耗了大量内存资源。
三、造成内存泄露的常见情形
- 集合类,比如 HashMap,ArrayList 等,这些对象经常会发生内存泄露。比如当它们被声明为静态对象时,它们的生命周期会跟应用程序的生命周期一样长,很容易造成内存不足。像 HashMap、Vector 等的使用最容易出现内存泄露,这些静态变量的生命周期和应用程序一致,他们所引用的所有的对象 Object 也不能被释放,因为他们也将一直被 Vector 等引用着。
Static Vector v = new Vector(10);
for (int i = 1; i<100; i++) {
Object o = new Object();
v.add(o);
o = null;
}
- 当集合里面的对象属性被修改后,再调用
remove()
方法时不起作用。
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;
}
}
-
监听器,在 java 编程中,我们都需要和监听器打交道,通常一个应用当中会用到很多监听器,我们会调用一个控件的诸如
addXXXListener()
等方法来增加监听器,但往往在释放对象的时候却没有记住去删除这些监听器,从而增加了内存泄漏的机会。 -
各种连接,比如数据库连接
(dataSourse.getConnection())
,网络连接 (socket) 和 io 连接,除非其显式的调用了其close()
方法将其连接关闭,否则是不会自动被 GC 回收的。对于 Resultset 和 Statement 对象可以不进行显式回收,但 Connection 一定要显式回收,因为 Connection 在任何时候都无法自动回收,而 Connection 一旦回收,Resultset 和 Statement 对象就会立即为 NULL。但是如果使用连接池,情况就不一样了,除了要显式地关闭连接,还必须显式地关闭 Resultset Statement 对象(关闭其中一个,另外一个也会关闭),否则就会造成大量的 Statement 对象无法释放,从而引起内存泄漏。这种情况下一般都会在 try 里面去的连接,在 finally 里面释放连接。 -
内部类和外部模块的引用,内部类的引用是比较容易遗忘的一种,而且一旦没释放可能导致一系列的后继类对象没有释放。此外程序员还要小心外部模块不经意的引用,例如程序员 A 负责 A 模块,调用了 B 模块的一个方法如:
public void registerMsg(Object b);
这种调用就要非常小心了,传入了一个对象,很可能模块 B 就保持了对该对象的引用,这时候就需要注意模块 B 是否提供相应的操作去除引用。
-
单例模式,不正确使用单例模式是引起内存泄漏的一个常见问题,单例对象在初始化后将在 JVM 的整个生命周期中存在(以静态变量的方式),如果单例对象持有外部的引用,那么这个对象将不能被 JVM 正常回收,导致内存泄漏,考虑下面的例子:
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 是个比较复杂的对象或者集合类型会发生什么情况。
四、内存泄露的解决方案
-
避免在循环中创建对象。
-
尽早释放无用对象的引用(最基本的建议)。
-
尽量少用静态变量,因为静态变量存放在永久代(方法区),永久代基本不参与垃圾回收。
-
使用字符串处理,避免使用 String,应大量使用 StringBuffer,每一个 String 对象都得独立占用内存一块区域。
作者:六尺帐篷
链接:https://www.jianshu.com/p/d2823693ccc2
网友评论