개발/Java

제네릭

na2ru2 2023. 7. 18. 17:16

제네릭은 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> {}

 

⚠️ 참고: [이것이 자바다] - 제네릭