귀찮지만 만들어보자

이펙티브 자바 - 메서드(4) 본문

이펙티브 자바

이펙티브 자바 - 메서드(4)

타우렌주술사 2019. 1. 27. 17:04

다중정의는 신중히 사용하라


public class CollectionClassfier {
  public static String classify(Set<?> s) {
    return "집합";
  }

  public static String classify(list<?> list) {
    return "리스트";
  }

  public static String classify(Collection<?> c) {
    return "그 외";
  }

  public static void main(String[] args) {
    Collection<?>[] collections = {
      new HashSet<String>(),
      new ArrayList<BigInteger>(),
      new HashMap<String, String>().values()
    };

    for (Collecstion<?> c: collections) {
      System.out.println(classify(c));
    }
  }
}

main문을 실행하면 집합 - 리스트 - 그 외 순서로 출력될것 같지만
실제로는 "그 외" 만 세번 출력 된다.
다중정의는 어느 메서드를 호출할 지 컴파일타임에 정해지기 때문이다.

재정의(override)한 메서드는 동적으로 선택되고, 다중정의(overload)한 메서드는 정적으로 선택된다.
재정의는 런타임에 '가장 하위에서 정의한' 메서드가 동작하는 방식이라서
상대적으로 다중정의에 비해 결과를 예상하기 쉽고, 유연한 방식이다.

public static String classify(Collection<?> c) {
  return c instanceof Set ? "집합" :
         c instanceof List ? "리스트" : "그 외";
}

코드를 이런식으로 바꾸면 다중정의를 사용하지 않으면서 원하는 결과를 불러올 수 있다.
이런 예제를 제공한 이유는 다중정의를 함부로 사용하지 말라는 뜻인것으로 보인다.

안전하고 보수적으로 가려면 매개변수 수가 같은 다중정의는 만들지 말자.
- 근데 상용 라이브러리 중에 이런 다중정의가 빈번하다.. 아마도 하위버전 지원때문이겠지..?
차라리 이럴 경우 메서드 이름을 다르게 주는게 더 나을 수 있다.

메서드를 다중정의할 때, 서로 다른 함수형 인터페이스라도 같은 위치의 인수로 받아서는 안된다.
- 서로 다른 함수형 인터페이스여도 서로 근본적으로 다르지 않기 때문이다

프로그래밍 언어가 다중정의를 허용한다고 해서 다중정의를 꼭 활용하라는 뜻은 아니다. 명심해두자

아 쓰다가 한번 날아감 딥빡