가변인수 메서드는 명시한 타입의 인수를 0개 이상 받을 수 있다. 가변인수 메서드를 호출하면, 가장 먼저 인수의 개수와 길이가 같은 배열을 만들고 인수들을 이 배열에 저장하여 가변인수 메서드에 건네준다.
간단한 가변인수 활용 예를 살펴보자.
//입력받은 int 인수들의 합을 계산해주는 가변인수 메서드다.
static int sum(int... args) {
int sum = 0;
for (int arg : args)
sum += arg;
return sum;
}
// sum(1,2,3)은 6을, sum()은 0을 돌려준다.
인수가 1개 이상이어야 할 때도 있는데, 예를 들어 최솟값을 찾는 메서드인데 인수를 0개만 받을 수도 있도록 설계하는 건 좋지 않다. 인수 개수는 런타임에 (자동 생성된) 배열의 길이로 알 수 있다.
인수가 1개 이상이어야 하는 가변인수 메서드를 구현한 예를 살펴보자.
//가변인자들 중 최솟값을 찾아 반환하는 메서드
static int min(int... args) {
if (args.length == 0) {
throw new IllegalArgumentException("인수가 1개 이상 필요합니다.");
}
int min = args[0];
for (int i = 0; i < args.length; i++)
if(args[i] < min)
min = args[i];
return min;
}
이 방식에는 문제가 몇 개 있다.
args
유효성 검사를 명시적으로 해야 하고, min
의 초깃값을 Integer.MAX_VALUE
로 설정하지 않고는 더 명료한 for-each
문도 사용할 수 없다.매개변수를 2개 받도록 하면 위의 문제가 말끔히 사라진다.
//가변인자들 중 최솟값을 찾아 반환하는 메서드
static int min(int firstArg, int... args) {
int min = firstArg;
for (int arg : args)
if (arg < min)
min = arg;
return min;
}
for-each
문이 쓰여 좀 더 깔끔하다.성능에 민감한 상황이라면 가변인수가 걸림돌이 될 수 있다. 가변인수 메서드는 호출될 때마다 배열을 새로 하나 할당하고 초기화한다. 큰 비용은 아닐 수도 있지만, 최적화하고 싶을 때가 있을 것이다.
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) { }