View
반응형
인수 개수가 일정하지 않은 메서드를 정의해야 한다면, 가변인수가 반드시 필요하다. 하지만 가변인수는 신중하게 사용해야 한다.
인수가 1개 이상이어야 하는 가변인수 메서드에서 인수를 0개 받을 수 있도록 설계하는 것은 좋지 않다. 인수 개수는 런타임에 배열의 길이로 알 수 있기 때문이다. 즉, 런타임에 실패할 수 있다는 뜻이다.
예를 들어, 최솟값을 찾는 가변인수 메서드가 있다고 생각해보자.
// 인수 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 유효성 검사를 명시적으로 해야 한다. min의 초기값을 Integer.MAX_VALUE로 설정하지 않고는 for-each문도 사용할 수 없다.
이 문제는 매개변수를 2개 받도록 강제하면 간단히 해결된다.
static int min(int firstArg, int... remainingArgs) {
int min = firstArg;
for (int i = 1; i < args.length; i++)
if (args[i] < min)
min = args[i];
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 a4, int... rest) {}
만약 메서드 호출의 95%가 인수를 3개 이하로 사용한다면, 마지막 5번째 다중정의 메서드에 가변인수를 추가해주자. 이렇게 하면 5%의 호출에서만 배열을 생성하기 때문에 특수한 상황에는 성능 최적화를 기대할 수 있게 된다.
이를 활용한 대표적인 예로는 EnumSet의 정적 팩터리가 있다. EnumSet의 정적 팩터리는 위 기법을 사용해 열거 타입 집합 생성 비용을 최소화한다.
핵심 정리
- 메서드를 정의할 때 필수 매개변수는 가변인수 앞에 두고, 가변인수를 사용할 때는 성능 문제까지 고려하자.
반응형
'BackEnd > 이펙티브 자바' 카테고리의 다른 글
[이펙티브 자바] Item55 - 옵셔널 반환은 신중히 하라 (0) | 2021.07.21 |
---|---|
[이펙티브 자바] Item54 - null이 아닌, 빈 컬렉션이나 배열을 반환하라 (0) | 2021.07.20 |
[이펙티브 자바] Item52 - 다중정의는 신중히 사용하라 (0) | 2021.07.09 |
[이펙티브 자바] Item51 - 메서드 시그니처를 신중히 설계하라 (0) | 2021.07.07 |
[이펙티브 자바] Item50 - 적시에 방어적 복사본을 만들라 (0) | 2021.06.30 |