맞는데 왜 틀릴까..?

Java/Effective Java

[Effective Java] Item 53. 가변인수는 신중히 사용하라

안도일 2023. 7. 16. 17:59
가변인수

 

가변인수 메서드는 명시한 타입의 인수를 0개 이상 받을 수 있다.

가변인수 메서드를 호출하면, 가장 먼저 인수의 개수와 길이가 같은 배열을 만들고 인수들을 이 배열에 저장하여 가변인수 메서드에 건네준다.

 

 

가변인수 활용 예

 

 

  • 입력받은 int 인수들의 합을 계산해 주는 가변인수 메서드다. 
  • sum(1,2,3)은 6을, sum()은 0을 돌려준다.

 

 

 

인수가 1개 이상이어야 하는 메서드 - 잘못 구현한 예

 

 

  • 지저분한 코드
  • 인수를 0개만 넣어 호출하면 컴파일타임이 아닌 런타임에 실패한다.
  • args 유효성 검사를 명시적으로 해야 하고, min의 초깃값을 Integer.MAX_VALUE로 설정하지 않고는 더 명료한 for-each문도 사용할 수 없다.

 

 

인수가 1개 이상이어야 하는 메서드 - 가변인수를 제대로 사용한 예

 

 

  • 첫 번째로 평범한 매개변수를 받고, 가변인수는 두 번째로 받으면 앞서의 문제가 해결된다.

 

 

 

가변인수는 인수 개수가 정해지지 않았을 때 아주 유용하다.

printf는 가변인수와 한 묶음으로 자바에 도입되어 가변인수의 효과를 보았다.

 

 

가변인수 성능 개선

 

성능에 민감한 상황이라면 가변인수가 걸림돌이 될 수 있다.

 

가변인수 메서드는 호출될 때마다 배열을 새로 하나 할당하고 초기화한다.

이 비용을 감당할 수는 없지만 가변인수의 유연성이 필요할 때 선택할 수 있는 패턴이 있다.

 

 

해당 메서드 호출이 95%가 인수를 3개 이하로 사용한다고 해보자.

 

 

  • 마지막 다중정의 메서드가 인수 4개 이상인 5%의 호출을 담당한다.
  • 따라서 메서드 호출 중 단 5%만이 배열을 생성하는데, 이 기법도 보통 때는 별 이득이 없지만, 꼭 필요한 특수 상황에서는 좋은 성능을 보여줄 것이다.
  • EnumSet의 정적 팩터리도 이 기법을 사용해 열거 타입 집합 생성 비용을 최소화한다.

 

 

결론

 

  • 인수 개수가 일정하지 않은 메서드를 정의해야 한다면 가변인수가 반드시 필요하다.
  • 메서드를 정의할 때 필수 매개변수는 가변인수 앞에 두고, 가변인수를 사용할 때는 성능 문제까지 고려하자.