<Effective Java> RULE 23 새 코드에는 무인자 제네릭 자료형을 사용하지 마라
자바 1.5이전 컬렉션
private final Collection stamps = ...;
// 엉뚱한 자료형을 넣어도 컴파일 시 문제가 없었다.
stamps.add(new Coin(...));
for(Iterator i = stamps.iterator() ; i.hasNext(); ){
Stamp s = (Stamp) i.next(); //ClassCastException 예외 발생
...//우표 작업
}
ClassCastException이 발생하게 되면 stamps컬렉션에 엉뚱한 객체를 넣는 코드를 찾아야 한다.
// 형인자 컬렉션 자료형 - 형 안전성 확보
private final Collection<Stamp> stamps = ...;
위와 같이 선언하면 stamps 컬렉션에 Stamp 객체만 넣을 수 있다는 것을 이해할 수 있으며 실제로 Stamp 객체만 삽입되도록 한다.
게다가 원소를 꺼낼 때 형변환을 하지 않아도 된다.
- for-each던 for문이건 마찬가지 이다.
List와 같은 무인자 자료형을 사용하면 형 안전성을 잃게 되지만, List<Object>와 같은 형인자 자료형을 쓰면 그렇지 않다.
// 실행 도중에 오류를 일으키는 무인자 자료형 사용 예
public static void main(String[] args){
List<String> strings = new ArrayList<String>();
unsafeAdd(strings, new Integer(42));
String s = strings.get(0); // 컴파일러가 자동으로 형 변환
}
private static void unsafeAdd(List list, Object o){
list.add(o);
}
//// 결과 : 컴파일은 잘되지만 무인자 자료형 list를 썻기 때문에 경고가 뜬다.
waring : unchecked call to add(E) in raw type List
list.add(o);
실제로 돌려보면 string.get(0)의 실행 결과를 String으로 변환하려 하기 때문에 ClassCastException예외가 발생한다.
버전 1.5부터 자바는 비한정적 와일드카드 자료형을 제공한다.
- 제네릭 자료형을 쓰고 싶으나 실제 형인자가 무엇인지는 모르거나 신경쓰고싶지 않을 때는 형인자로 ?를 쓰면 된다.
- Set<E>에 대한 비한정적 와일드 카드 자료형은 Set<?>이다.
//비한정적 와일드 카드 자료형 - 형 안전성과 유연성 만족
static int numElementsIncommon(Set<?> s1, Set<?> s2){
int result = 0;
for (Object o1 :s1){
if(s2.contains(o1))
result++;
return result;
}
Set<?>과 Set의 차이는 무엇인가?
- 와일드 카드 자료형은 안전하지만 무인자 자료형은 그렇지 않다.
- 무인자 자료형은 아무 객체나 넣을 수 있어서 컬릭션 자료형 불변식이 깨진다.
- 그러나 Collection<?>에는 null이외에 어떤 원소도 넣을 수 없다.
ㄴ어떤 자료형의 객체를 담는 컬렉션인지 알 방법이 없기 때문이다.
새로운 코드에는 무인자 자료형을 쓰면 안된다고 했지만 예외가 두가지 있다.
1) 클래스 리터럴에는 반드시 무인자 자료형을 사용해야 한다.
- 자바 표준에 따르면 클래스 리터럴에는 형인자 자료형을 쓸 수 없다.
ㄴ 배열 자료형이나 기본자료형은 쓸수 있다.
ㄴex) List.class, String[].class, int.class...
- instanceof 연산자는 비한정적 와일드 카드 자료형 이외의 형인자 자료형에 적용할 수 없다.
// instancof 연산자에는 무인자 자료형을 써도 OK
if( o instanceof Set ){ //무인자 자료형
Set<?> m = (Set<?>) o; //와일드 카드 자료형
....
}
* o가 Set객체라는것이 확실해진 다음에는 와일드 카드 자료형 Set<?>로 형변환 해야하는 것에 주의하자.
요약
- 무인자 자료형을 쓰면 프로그램 실행 도중에 예외가 발생할 수 있으므로, 새로 만드는 코드에는 무인자 자료형을 쓰지 말아야 한다.
- 무인자 자료형을 제네릭 도입 전 코드 호환성을 유지하기 위해 제공하는것에 불과하다.
- Set<Object> 는 어떤 자료형의 객체도 담을 수 있는 집합의 자료형
- Set<?>는 모종의 자료형 객체만 담을 수 있는 집합의 자료형
- Set<Object>와 Set<?>는 안전하지만 Set은 안전하지 않다.
용어
|
예시
|
형인자 자료형
|
List<String>
|
실 형인자
|
String
|
제네릭 자료형
|
List<E>
|
형식 형인자
|
E
|
비한정적 와일드카드 자료형
|
List<?>
|
무인자 자료형
|
List
|
한정적 형인자
|
<E extends Number>
|
재귀적 형 한정
|
<T extends Comparable<T>>
|
한정적 와일드카드 자료형
|
List<? extends Number>
|
제네릭 메서드
|
static <E> List<E> asList(E[] a)
|
자료형 토큰
|
String.class
|