// 코드 7-1 메모리 누수가 일어나는 위치는 어디인가? (36쪽)
public class Stack {
private Object[] elements;
private int size = 0;
private static final int DEFAULT_INITIAL_CAPACITY = 16;
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);
}
// // 코드 7-2 제대로 구현한 pop 메서드 (37쪽)
// public Object pop() {
// if (size == 0)
// throw new EmptyStackException();
// Object result = elements[--size];
// elements[size] = null; // 다 쓴 참조 해제
// return result;
// }
public static void main(String[] args) {
Stack stack = new Stack();
for (String arg : args)
stack.push(arg);
while (true)
System.err.println(stack.pop());
}
}
- 스택이 자기 메모리를 직접 관리하기 때문이다.
- 배열의 활성 영역에 속한 원소들이 사용되고 비활성 영역은 쓰이지 않는다.
- 활성 영역 : 인덱스가 size보다 작은 원소들로 구성된다. 그 외는 비활성 영역
- 문제는 위 사실을 가비지 컬렉터는 알 길이 없다. 그러므로 null 처리로 가비지 컬렉터에게 알려줘야 한다.
- 외부에서 키를 참조하는 동안만 엔트리가 살아 있다
- ScheduledThreadPoolExecutor : 쓰지 않는 엔트리 청소
- LinkedHashMap은
removeEldestEntry()
메소드를 사용하여 엔트리 청소
java.lang.ref
참고Reference: