본문은 Effective Java를 읽고 간단하게 정리한 글입니다. 필요에 따라 생략/수정된 부분이 있을 수 있으며, 내용이 추후 변경될 수 있습니다.
선결론
- 인수 개수가 일정하지 않은 메서드를 정의해야 하는 경우엔 가변인수를 사용한다
- 메서드를 정의할 때 필수 매개변수는 가변인수 앞에 둔다
- 가변인수를 사용할 때는 성능 문제까지 고려하자
가변인수 메서드
- 가변인수 메서드는 명시한 타입의 인수를 0개 이상 받을 수 있다
- 가변인수 메서드를 호출하면, 가장 먼저 인수의 개수와 길이가 같은 배열을 만들고 인수들을 이 배열에 저장하여 가변인수 메서드에 건네준다(아이템 32)
간단한 가변인수 활용 예
static int sum(int... args) {
int sum = 0;
for (int arg : args) // 배열에서 하나씩 꺼낸다
sum += arg;
return sum;
}
인수가 1개 이상이어야 할 때는 다음과 같이 설계할 수 있다. 아래 코드는 잘못 구현한 예로써 아래와 같은 문제를 가진다.
static int min(int... args) {
if (args.length == 0)
throw new IllegalArgumentException("인수가 1개 이상 필요합니다.");
int min = args[0];
for (int i = 1; i < args.length; i++)
if (args[i] < min)
min = args[i];
return min;
}
- 인수를 0개만 넣어 호출하면 런타임에 실패한다
- 코드도 지저분하다
- args 유효성 검사를 명시적으로 해야 한다
매개변수를 2개 받도록 하면 더 간단히 구현할 수 있다.
static int min(int firstArgs, int... remainingArgs) {
int min = firstArg;
for (int arg : remainingArgs)
if (arg < min)
min = arg;
return min;
}
- 첫 번째로는 평범한 매개변수를 받고, 가변인수는 두 번째로 받으면 앞의 문제를 해결할 수 있다
성능에 민감한 상황에 사용할 수 있는 패턴
public void foo() { }
public void foo(int a1) { }
public void foo(int a1, int a2) { }
public void foo(int a1, int a2, int a3) { }
public void foo(int a1, int a2, int a3, int… rest) { }
- 가변 인수 메서드는 호출될 때마다 배열을 새로 하나 할당하고 초기화한다
- 따라서 성능에 민감한 상황이라면 가변인수가 걸림돌이 될 수 있다
- 다중정의를 활용하여 최적화할 수 있다
- 호출 시 인수가 0개 ~ 3개인 경우엔 배열을 할당-초기화하는 가정이 없기 때문이다
- EnumSet의 정적 팩터리도 이 기법을 사용해 열거 타입 집합 생성 비용을 최소화한다
'책 > Effective Java' 카테고리의 다른 글
[이펙티브 자바] 아이템 55: 옵셔널 반환은 신중히 하라 (0) | 2022.06.29 |
---|---|
[이펙티브 자바] 아이템 54: null이 아닌 빈 컬렉션이나 배열을 반환하라 (0) | 2022.06.23 |
[이펙티브 자바] 아이템 51: 메서드 시그니처를 신중히 설계하라 (0) | 2022.06.22 |
[이펙티브 자바] 아이템 50: 적시에 방어적 복사본을 만들라 (0) | 2022.06.21 |
[이펙티브 자바] 아이템 48: 스트림 병렬화는 주의해서 적용하라 (0) | 2022.06.17 |