Friday, July 17, 2015

Effective Java item 6: Eliminate obsolete object references

In Effective Java book, item 6, there is a source code that causes memory leak.

I tested the code to see if it really causes critical memory leak, and here is the result




The source code from the book that causes memory leak.

public class Stack {
    private Object[] elements;
    private int size = 0;
    private static final int DEFAULT_INITIAL_CAPACITY = 1;
    
    public Stack() {
        elements = new Object[DEFAULT_INITIAL_CAPACITY];
    }
    
    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) 
            elements = Arrays.copyOf(elements, 2*size + 1);
    }
}

The part that causes memory leak is pop() method.

When the stack grows and then shrinks, the stack still holds the obsolete object references.

To avoid that, we need to null out the obsolete objects.

Here is the modified code of pop() method.

public Object pop(){
    if(size==0){
        throw new EmptyStackException();
    }
    
    Object obj = elements[--size];
    elements[size] = null;
    return obj;
}


Let's test if the original code really causes memory leak, and the modified one solves the problem.

Here is the code that I used to test. First test uses the original Stack class that causes memory leak.

public class TestApplication {

    /**
     * @param args the command line arguments
     */
    public static void main(String[] args) {
        
        long startTime = System.currentTimeMillis();
           
        Stack stack = new Stack();
        
        for(int j=0; j<10000000; j++) {
            stack.push(new Object());
        }
        
        for(int j=0; j<10000000; j++) {
            stack.pop();
        }

        // Get the Java runtime
        Runtime runtime = Runtime.getRuntime();
        // Run the garbage collector
        runtime.gc();
        // Calculate the used memory
        long memory = runtime.totalMemory() - runtime.freeMemory();
        System.out.println("Used memory is bytes: " + memory);
        System.out.println("Used memory is megabytes: "
            + bytesToMegabytes(memory));
       
        
        long stopTime = System.currentTimeMillis();
        long elapsedTime = stopTime - startTime;
        System.out.println("elapsed Time is: " + elapsedTime);
        
            
    }
    
    private static final long MEGABYTE = 1024L * 1024L;

    public static long bytesToMegabytes(long bytes) {
      return bytes / MEGABYTE;
    }

}
Result is
Used memory is bytes: 227572000
Used memory is megabytes: 217
elapsed Time is: 5279
Now modify the Stack class so that no memory leak created, and the Result is
Used memory is bytes: 67572000
Used memory is megabytes: 64
elapsed Time is: 3724


Clearly, the pop() method had memory leak problem, and by null out the obsolete object, it is solved

Done

No comments:

Post a Comment