所谓内存泄露就是指一个不再被程序使用的对象或变量一直被占据在内存中。Java中有垃圾回收机制,它可以保证一对象不再被引用的时候,即对象编程了孤儿的时候,对象将自动被垃圾回收器从内存中清除掉。由于Java 使用有向图的方式进行垃圾回收管理,可以消除引用循环的问题,例如有两个对象,相互引用,只要它们和根进程不可达的,那么GC也是可以回收它们的。
1 package com.huawei.interview; 2 3 import java.io.IOException; 4 5 public class GarbageTest { 6 /** 7 * @param args 8 * @throws IOException 9 */ 10 public static void main(String[] args) throws IOException { 11 // TODO Auto-generated method stub 12 try { 13 gcTest(); 14 } catch (IOException e) { 15 // TODO Auto-generated catch block 16 e.printStackTrace(); 17 } 18 System.out.println("has exited gcTest!"); 19 System.in.read(); 20 System.in.read(); 21 System.out.println("out begin gc!"); 22 for(int i=0;i<100;i++) 23 { 24 System.gc(); 25 System.in.read(); 26 System.in.read(); 27 } 28 } 29 30 private static void gcTest() throws IOException { 31 System.in.read(); 32 System.in.read(); 33 Person p1 = new Person(); 34 System.in.read(); 35 System.in.read(); 36 Person p2 = new Person(); 37 p1.setMate(p2); 38 p2.setMate(p1); 39 System.out.println("before exit gctest!"); 40 System.in.read(); 41 System.in.read(); 42 System.gc(); 43 System.out.println("exit gctest!"); 44 } 45 46 private static class Person 47 { 48 byte[] data = new byte[20000000]; 49 Person mate = null; 50 public void setMate(Person other) 51 { 52 mate = other; 53 } 54 } 55 }
java中的内存泄露的情况:长生命周期的对象持有短生命周期对象的引用就很可能发生内存泄露,尽管短生命周期对象已经不再需要,但是因为长生命周期对象持有它的引用而导致不能被回收,这就是java中内存泄露的发生场景,通俗地说,就是程序员可能创建了一个对象,以后一直不再使用这个对象,这个对象却一直被引用,即这个对象无用但是却无法被垃圾回收器回收的,这就是java中可能出现内存泄露的情况,例如,缓存系统,我们加载了一个对象放在缓存中(例如放在一个全局map对象中),然后一直不再使用它,这个对象一直被缓存引用,但却不再被使用。
检查java中的内存泄露,一定要让程序将各种分支情况都完整执行到程序结束,然后看某个对象是否被使用过,如果没有,则才能判定这个对象属于内存泄露。
如果一个外部类的实例对象的方法返回了一个内部类的实例对象,这个内部类对象被长期引用了,即使那个外部类实例对象不再被使用,但由于内部类持久外部类的实例对象,这个外部类对象将不会被垃圾回收,这也会造成内存泄露。
public class Stack {
private Object[] elements=new Object[10];
private int size = 0;
public void push(Object e){
ensureCapacity();
elements[size++] = e;
}
public Object pop(){
if( size == 0)
throw new EmptyStackException();
return elements[–size];
}
private void ensureCapacity(){
if(elements.length == size){
Object[] oldElements = elements;
elements = new Object[2 * elements.length+1];
System.arraycopy(oldElements,0, elements, 0, size);
}
}
}
内存泄露的另外一种情况:当一个对象被存储进HashSet集合中以后,就不能修改这个对象中的那些参与计算哈希值的字段了,否则,对象修改后的哈希值与最初存储进HashSet集合中时的哈希值就不同了,在这种情况下,即使在contains方法使用该对象的当前引用作为的参数去HashSet集合中检索对象,也将返回找不到对象的结果,这也会导致无法从HashSet集合中单独删除当前对象,造成内存泄露