publicclassStack {privateObject[] elements;privateint size =0;privatestaticfinalint DEFAULT_CAPACITY =16;publicStack() { elements =newObject[DEFAULT_CAPACITY]; }publicvoidpush(Object e) {ensureCapacity(); elements[size++] =0; }publicObjectpop() {if (size ==0) {thrownewEmptyStackException(); }return elements[--size]; }// 원소를 위한 공간을 적어도 하나 이상 여유를 두며, 늘려야하는 경우 두배 이상 늘린다.privatevoidensureCapacity() {if (elements.length== size) { elements =Arrays.copyOf(elements,2* size +1); } }}
제네릭 타입 안에서 리스트를 사용하는 것이 항상 가능한 것도, 좋은 것도 아니다. 자바가 리스트를 기본타입으로 제공하지 않아, ArrayList 와 같은 제네릭 타입도 결국은 기본 타입인 배열을 사용해 구현해야하며, HashMap 의 경우 성능을 높일 목적으로 배열을 사용하기도 한다.
대부분 제네릭 타입은 타입 매개변수에 아무런 제약을 두지 않으며, Stack<Object>, Stack<int[]>, Stack<List<String>>, Stack 등 어떤 참조 타입으로도 생성할 수 있다. 단, 기본타입은 사용할 수 없다.Stack<int> 같이 기본타입으로 만들려고 하면 컴파일 오류가 발생한다. 해당 오류는 자바 제네릭 타입 시스템의 근본적인 문제이며, item61-박싱된 기본타입을 사용해 우회할 수 있다.
추가적으로, 한정적 타입 매개변수(bounded type parameter)를 사용해 매개변수에 제약을 둘 수도 있다.
<E extends Delayed>는 Delayed 하위 타입만 받는다는 뜻이며, DelayQueue 의 원소에서 형변환 없이 바로 Delayed 메서드를 사용할 수 있다. 또한, ClassCastException 오류도 걱정할 필요가 없다.
public class Stack<E> {
// private으로 저장
private E[] elements;
private int size = 0;
private static final int DEFAULT_INITIAL_CAPACITY = 15;
public Stack() {
elements = (E[]) new Object[DEFAULT_INITIAL_CAPACITY];
}
public void push(E e) {
ensureCapacity();
elements[size++] = e;
}
public E pop() {
if (isEmpty()) {
throw new EmptyStackException();
}
E result = elements[size--];
elements[size] = null;
return result;
}
public boolean isEmpty() {
return size == 0;
}
private void ensureCapacity() {
if (elements.length == size) {
elements = Arrays.copyOf(elements, 2 * size + 1);
}
}
}
Unchecked cast: 'java.lang.Object[]' to 'E[]'
// elements 배열은 push(E)로 넘어온 E인스턴스만 담는다.
// 타입 안정성을 보장하지만, 런타임 타입은 E[]가 아닌 Object[]이다.
@SuppressWarnings("unchecked")
public Stack() {
elements = (E[]) new Object[DEFAULT_INITIAL_CAPACITY];
}
public class Stack<E> {
// private으로 저장
private Object[] elements;
private int size = 0;
private static final int DEFAULT_INITIAL_CAPACITY = 15;
public Stack() {
elements = new Object[DEFAULT_INITIAL_CAPACITY];
}
public void push(E e) {
ensureCapacity();
elements[size++] = e;
}
public E pop() {
if (isEmpty()) {
throw new EmptyStackException();
}
E result = elements[size--];
elements[size] = null;
return result;
}
public boolean isEmpty() {
return size == 0;
}
private void ensureCapacity() {
if (elements.length == size) {
elements = Arrays.copyOf(elements, 2 * size + 1);
}
}
}
java: incompatible types: java.lang.Object cannot be converted to E
E result = (E) elements[size--];
Unchecked cast: 'java.lang.Object' to 'E'
// push에서 E타입만 허용하므로 안전
@SuppressWarnings("unchecked")E result = (E) elements[size--];
Stack<String> stack = new Stack<>();
public class DelayQueue<E extends Delayed> extends AbstractQueue<E>
implements BlockingQueue<E> {