인수 개수가 일정하지 않은 메서드를 정의해야 한다면 가변인수가 반드시 필요하다. 메서드를 정의할 때 필수 매개변수는 가변인수 앞에 두고, 가변인수를 사용할 때는 성능 문제까지 고려하자.
가변인수 메서드를 호출하면, 가장 먼저 인수의 개수와 길이가 같은 배열을 만들고 인수들을 이 배열에 저장하여 가변인수 메서드에 건네준다.
ex) 간단한 가변인수 활용 예
static int sum(int... args) {
int sum = 0;
for (int arg : args) {
sum += arg;
}
return sum;
}
ex) 인수가 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개만 넣어도 min 함수가 호출하면 컴파일 타임이 아닌 런타임에 실패한다.
- 코드도 지저분하다.
- args 유효성 검사를 명시적으로 해야 하고, min의 초기값을 Integer.MAX_VALUE로 설정하지 않고는 for-each 문도 사용할 수 없다.
- 첫 번째로 평범한 매개변수를 받고, 가변인수를 두 번째로 받으면 해당 문제는 해결된다.
ex) 인수가 1개 이상이어야 할 때 가변인수를 제대로 사용하는 방법
static int min(int firstArg, int... remainingArgs) {
int min = firstArg;
for (int arg : remainingArgs) {
if (arg < min) {
min = arg;
}
}
rutnr 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) {}
또한 EunumSet의 정적 팩터리도 이 기범을 사용해 열거 타입 집합 생성 비용을 최소화한다.
EumSet은 비트 필드를 대체하면서 성능까지 유지해야 하므로 아주 적절하게 활용한 에라 할 수 있다.
'Java > EffectiveJava' 카테고리의 다른 글
Effective Java Item54. null이 아닌, 빈 컬렉션이나 배열을 반환하라. (1) | 2025.02.22 |
---|---|
Effective Java Item42. 익명 클래스보다는 람다를 사용하라. (1) | 2025.02.16 |
Effective Java Item41. 정의하려는 것이 타입이라면 마커 인터페이스를 사용하라. (0) | 2025.02.16 |
Effective Java Item33. 타입 안전 이종 컨테이너를 고려하라. (1) | 2025.02.09 |
Effective Java Item32. 제네릭과 가변인수를 함께 쓸 때는 신중하라. (0) | 2025.02.09 |