- 타입 매개변수(Type Parameter)를 이용하여 클래스, 인터페이스, 메서드를 정의할 때 타입을 일반화시켜주는 기능.
- 코드의 유연성과 재사용성을 높이고, 컴파일 시 타입 안전성을 보장함.
- 타입 언어에서 “중복되거나 필요없는 코드를 줄여주는 것”
여기서 중복되는 코드가 의미하는게 뭘까?
function plusReturnFunction(a, b) {
return a + b;
}
const a = 1;
const b = 2;
const c = 1.1;
const d = "hello";
plusReturnFunction(a + b); // 3
plusReturnFunction(a + c); // 2.1
plusReturnFunction(a + d); // 1hello
자바스크립트를 예로들어 설명하면
기본적으로 타입을 지정 해 줄 필요가 없기에 하나의 함수만 구현하면 된다
public class Generic {
public String plusReturnFunction(int a, int b) { ... }
public String plusReturnFunction(int a, long b) { ... }
public String plusReturnFunction(int a, String b) { ... }
}
하지만 자바는 일반적으로 타입을 지정해줘야한다
그래서 똑같은 로직을 수행하는 함수를 타입을 지정해야 한다는 이유로 세 차례나 구현해야 한다
- 그러면서도 타입 안정성을 해치지 않는 것
그렇다면 타입의 안정성을 해치는 것은 뭘까?
public class Generic {
public Object plusReturnFunction(Object a,Object b) { ... }
}
public class Generic<T> { ... }
Generic<String> stringGeneric = new Generic<>();
- Generic의 클래스처럼, 제네릭을 사용한 클래스를 제네릭 클래스라고 한다
- 제네릭에서 <>사이에 들어가는 변수명 T는 타입 변수라고 한다
- Generic 클래스를 원시 타입 이라고 한다
public class Generic<T, U, E> {
public E multiTypeMethod(T t, U u) { ... }
}
Generic<Long, Integer, String> instance = new Generic();
instance.multiTypeMethod(longVal, intVal);
- 다형성, 즉 상속과 타입의 관계는 그대로 적용된다
- 대표적으로 부모 클래스로 제네릭 타입변수를 지정하고, 그 안에 자식클래스를 넘기는 것은 잘 동작한다
- 제네릭에서 와일드카드(Wildcard)는 특정 타입을 대체할 수 있는 기호
- 주로 메서드의 매개변수나 반환 타입으로 사용되며, 불특정한 타입을 나타낸다.
- 와일드카드는 ? 기호로 표현
public class ParkingLot<T extends Car> { ... }
ParkingLot<BMW> bmwParkingLot = new ParkingLot();
ParkingLot<Iphone> iphoneParkingLot = new ParkingLog(); // error!
1.상한 와일드카드(Upper Bounded Wildcards) <? extends T>
: T와 그 자손들만 사용 가능
2.하한 와일드카드(Lower Bounded Wildcards)<? super T>
: T와 그 조상들만 가능
3.와일드카드(Wildcard)<?>
: 제한 없음
이렇게 제한을 하는 이유는 다형성 때문이다
위의 코드에서, T는 Car의 자손클래스들이라고 정의했기 때문에,
해당 클래스 내부에서 최소 Car 객체에 멤버를 접근하는 코드를 적을 수 있다
반대로 그러한 코드들이 있을 여지가 있기 때문에,
Car 객체의 자손이 아닌 클래스는 제한한다

// 또는 ..
static <T> void sort(List<T> list, Comparator<? super T> c) { ... }
public class Generic<T, U, E> {
// Generic<T,U,E> 의 T와 아래의 T는 이름만 같을뿐 다른 변수
static <T> void sort(List<T> list, Comparator<? super T> c) { ... }
}