티스토리 뷰

728x90
반응형

가변인수는 신중히 사용하라


💡 가변인수란?

  • 매개변수를 동적으로 받을 수 있는 방법이며, 자바 5부터 지원되기 시작했습니다.
  • 매개변수에 ...을 사용하여 가변인수를 설정할 수 있고, 0개 이상의 인수를 받을 수 있습니다.
  • 가변인수 메서드를 호출하면 가장 먼저 인수의 개수와 길이가 같은 배열이 만들어지고 생성된 배열에 인수들을 저장하여 메서드에 전달하며, 해당 메서드에서는 생성된 배열을 사용합니다.
  • 이처럼 메서드를 호출할 때마다 새로운 배열을 생성해서 전달해주는데, 이 부분도 결국 비용이고 낭비일 수도 있습니다.
public class Example {

    public static void main(String[] args) {

        varargs("H", "E", "L", "L", "O");
    }

    public static void varargs(String... strings) {
        System.out.println(Arrays.toString(strings));
        for (String s : strings) {
            System.out.println(s);
        }
    }
}

 

💡 문제가 발생하는 가변인수

  • 아래 코드는 최솟값을 구하는 메서드인데 가변인수를 매개변수로 하는것은 좋지 않습니다. 매개변수를 아무것도 넣지 않아도 오류가 발생하지 않고 런타임시점에서야 오류가 발생합니다. 오류는 최대한 빨리 컴파일 시점에 아는것이 좋습니다.
  • 또한 배열의 길이를 체크하고 유효성 검사를 해야하니 코드도 지저분해질 수 있습니다.
public class Example {

    public static void main(String[] args) {
        minValue();
    }

    public static void minValue(int... numbers) {

        if (numbers.length == 0) {
            throw new IllegalArgumentException();
        }

        int min = numbers[0];
        for (int n : numbers) {
            if (n < min) {
                min = n;
            }
        }
        System.out.println(min);
    }
}

 

💡 해결방법

  • 아래 예제코드처럼 첫번째 매개변수를 최솟값으로 지정하고, 가변인수 배열을 비교해 최솟값을 구할 수 있습니다.
public class Example {

    public static void main(String[] args) {

        minValue(1, 2, 3, 4, 5, 6);
    }

    public static void minValue(int first, int... numbers) {
        int min = first;
        for (int n : numbers) {
            if (n < min) {
                min = n;
            }
        }
        System.out.println(min);
    }
}

 

💡 성능 문제와 해결 방안

  • 가변인수는 매개변수의 개수가 정해지지 않은 경우 유용하게 사용할 수 있습니다. 하지만 메서드를 호출할 때마다 매 번 새로운 배열을 생성해야하고 값을 할당해야하는 비용적인 문제가 발생할 수 있습니다. 그렇기 때문에 가변인수를 무작정 쓰기보다는 메서드의 호출 패턴을 분석하여 오버로딩을 활용할 수도 있습니다.
  • 만약 가변인수를 받는 메서드에서 메서드 호출의 90% 이상이 인수의 개수를 3개 이하로 받을 경우 아래와 같이 다중정의를 이용해 배열이 새로 생성되는 비율을 줄일 수 있습니다.
public void method(){ ... }
public void method(int a1){ ... }
public void method(int a1, int a2){ ... }
public void method(int a1, int a2, int a3){ ... }
public void method(int a1, int a2, int a3, int... args){ ... }

 

 

 

 

 

728x90
반응형