제네릭
제네릭은 Java 5부터 추가된 타입이다.
제네릭 타입을 이용하면 잘못된 타입이 사용될 수 있는 문제를 컴파일 과정에서 발견할 수 있다.
제네릭은 다음과 같은 이점을 가지고 있다.
1. 컴파일 시 강한 타입 체크가 가능해진다.
2. 타입 변환이 제거된다.
제네릭 타입 (class<T>, interface<T>)
타입을 파라미터로 가지는 클래스나 인터페이스를 말한다. 제네릭 타입은 이름 뒤에 <> 부호가 붙고 <> 사이에 타입 파라미터가 위치한다.
제네릭을 사용한 코드와 사용하지 않은 코드를 비교해 보면서 차이를 알아보자.
public class Box {
private Object object;
public void set(Object object) {
this.object = object;
}
public Object get() {
return object;
}
}
...
Box box = new Box();
// String 타입을 Object 타입으로 자동 변환해서 설정
box.set("hello");
// Object 타입을 String 타입으로 강제 타입 변환해서 가져옴
String str = (String) box.get();
제네릭을 사용하지 않은 코드에서는 타입변환이 일어나는 모습을 확인할 수 있다.
public class Box<T> {
private T t;
public void set(T t) {
this.t = t
}
public T get() {
return this.t;
}
}
...
Box<String> box = new Box<String>();
box.set("Hello");
String str = box.get();
타입 변환이 제거된 모습을 확인할 수 있다.
멀티 타입 파라미터 (class<K,V,...>, interface<K,V,....>)
제네릭 타입은 두 개 이상의 멀티 타입 파라미터를 사용할 수 있다.
public class Box<A, B> {
private A name;
private B age;
public void setName(A name) {
this.name = name;
}
public void setAge(B age) {
this.code = age;
}
public A getName() {
return this.name;
}
public B getAge() {
return this.age;
}
}
...
Box<String, Integer> box = new Box<String, Integer>();
box.setName("홍길동");
box.setAge(25);
String name = box.getName();
int age = box.getAge();
제네릭 메소드 (<T, R> R method(T t))
매개 타입과 리턴 타입으로 타입 파라미터를 갖는 메소드를 뜻한다.
public <타입 파라미터> 리턴 타입 메소드명(매개변수) {}
제네릭 메소드 호출은 다음과 같이 할 수 있다.
// 명시적으로 구체적인 타입을 지정
리턴타입 변수 = <구체적 타입> 메소드명(매개값)
// 매개 값으로 묵시적인 타입 추정
리턴타입 변수 = 메소드명(매개값)
제네릭 메소드가 사용되는 예
public class Box<T> {
private T value;
public void set(T value){
this.value = value;
}
public T get() {
return this.value;
}
}
public class Util {
public static <T> Box<T> boxing(T t) {
Box<T> box = new Box(T);
box.set(t);
return box;
}
}
public class BoxingMethodExample {
public static void main(String[] args) {
// 명시적 호출
Box<Integer> box1 = Util.<Integer>boxing(100);
int intVlaue = box1.get();
// 묵시적 호출
Box<String> box2 = Util.boxing("홍길동");
String strValue = box2.get();
}
}
제한된 타입 파라미터 (<T extends 최상위 타입>)
타입 파라미터에서 지정되는 구체적인 타입을 제한하기 위해 사용된다.
제한된 타입 파라미터 메소드는 다음과 같이 선언할 수 있다.
public <T extends 상위타입> 리턴타입 메소드(매개변수) {}
제한된 타입 파라미터가 사용되는 예
public class Util {
public static <T extends Number> int compare(T t1, T t2) {
// 매개변수로 넘어온 값을 double 값으로 저장
double v1 = t1.doubleValue();
double v2 = t2.doubleValue();
// Double.compare는 첫 번째 매개 값이 작으면 -1을, 같으면 0을, 크면 1을 리턴하는 메소드이다.
return Double.compare(v1, v2);
}
}
public class BoundedTypeParameterExample {
public static void main(String[] args) {
// 에러: String은 Number의 하위타입이 아님
// String str = <String>Util.compare("안녕", "하세요");
int result1 = <Integer>Util.compare(10, 20);
int result2 = <Double>Util.compare(4.5, 3);
}
}
와일드 카드 타입 (<?>, <? extends ...>, <? super ...]>)
제네릭 타입을 구체적인 타입 대신 와일드카드로 사용하고 싶을 때 다음과 같은 세 가지 방법으로 사용할 수 있다.
1. 제네릭 타입 <?>: 제한 없음 - 모든 클래스나 인터페이스 타입이 올 수 있다.
2. 제네릭 타입 <? extends 상위타입>: 상위클래스 제한 - 상위타입이나 하위타입만 올 수 있다.
3. 제네릭 타입 <? super 하위타입>: 하위클래스 제한 - 하위타입이나 상위타입이 올 수 있다.
다음 예로 쉽게 이해해 보자.
Person의 하위 클래스로 Worker와 Student가 있고, Student의 하위 클래스로 HighStudent가 있다.
1. Course<?>
수강생은 모든 타입(Person, Worker, HighStudent)이 될 수 있다.
2. Course<? extends Student>
수강생은 Student와 HighStudent만 될 수 있다.
3. Course<? super Worker>
수강생은 Worker와 Person만 될 수 있다.
제네릭 타입의 상속과 구현
제네릭 타입도 부모 클래스가 될 수 있다.
다음 예는 제네릭 클래스를 상속받은 클래스의 예다.
// 자식 제네릭 타입은 추가적인 타입 파라미터를 가질 수 있다.
public class ChildProduct<A, B, C> extends Product<T, M> {}
⚠️ 참고: [이것이 자바다] - 제네릭