从Java1.5开始就增加了可变参数(varargs)方法,又称作variable arity method。可变参数方法接受0个或多个指定类型的参数。它的机制是先创建一个数组,数组的大小为调用位置所传递的参数数量,然后将值传到数组中,最后将数组传递到方法。
例如下面有个例子,返回多个参数的和:
static int sum(int... args) {
int sum =0;
for(int arg : args)
sum += arg;
return sum;
}
很多时候,我们需要至少一个参数,那么很容易想到在方法开始的时候做参数检查,下面是一个计算参数最小值的例子:
static int min(int... args) {
if(args.length ==0)
throw new Illegal Argument Exception("Too few arguments");
intmin = args[0];
for(inti =1; i < args.length; i++)
if(args[i] < min)
min = args[i];
return min;
}
以上是在方法开始的时候检查参数长度是否为0。但是,这是解决方案有两个不足:1.如果没有传入参数,只有在运行的时候失败,而不是编译的时候失败;2.代码不美观,除了需要在最开始检查有效性之外,在这个案例中,比较参数的大小的时候,只能从数组第二个开始比较,代码不够简洁美观。
很巧的是,利用可变参数的语法,正好有一种巧妙的方法可以解决这个问题:声明该方法有两个参数,一个是指定类型的正常参数,另一个是这种类型的varargs参数。这个方法弥补了上面的不足(不需要再检查参数的数量了,因为至少要传递一个参数,否则不能通过编译):
static int min(int firstArg , int... remainingArgs) {
intmin = firstArg;
for(intarg : remainingArgs)
if(arg < min)
min = arg;
return min;
}
事实上,当你真的需要让一个方法带有不定数量的参数的时候,可变参数才会变得非常有效。它本来是为printf 和反射机制(见53条)设定的。
接下来让我们一起看看一个有趣的例子:
List homophones = Arrays.asList("to","too","two");
System.out.println(homophones);
int[] digits = {1,2,3,4,5};
System.out.println(Arrays.asList(digits));
输出结果是:
[to, too, two]
[[I@15db9742]
在以上的这个例子中,System.out.println调用的是toString,而List是从Object继承了它们的toString实现。如果使用asList方法来初始化int数组,它会忠实的将int数组包装到List实例中,打印这个List会导致到List中调用toString,toString的是int[],打印的是数组地址,得到我们并不想看到的结果。
List list=Arrays.asList(digits);
我将代码稍作修改,更能说明这个问题:
List homophones = Arrays.asList("to","too","two");
System.out.println(homophones);
int[] digits = {1,2,3,4,5};
List list=Arrays.asList(digits);
System.out.println(list);
String[] strs={"to","too","two"};
System.out.println(strs);
System.out.println(digits);
输出结果是:
[to, too, two]
[[I@15db9742]
[Ljava.lang.String;@6d06d69c]
[I@15db9742]
使用Arrays.toString(digits);方法就可以避免这个问题,专门将任何类型的数组转换成字符串而设计
另外,需要注意的是,在重视性能的情况下,使用可变参数机制要特别小心。可变参数方法每次调用都会导致进行一次数组分配和初始化。如果只是凭经验确定,无法承受这一成本,但是又需要可变参数的灵活性。这时候,需要评估,假如某个方法95%会调用3个或更少的参数,那么就声明该方法的5个重载(和上一条一样,这几个重载的方法必须尽量保证方法的功能相同,返回值相同),每个重载方法带有0-3个参数,超过3个参数的时候,就会自动调用可变参数方法。
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){}
像大多数的性能优化方法一样,这种方式看起来很不恰当,但是用到的时候会有很大帮助。
总之,和其他规则一样,尽管可变参数是一个很方便的方式,但是它们不应该被过度滥用。除非有必要,尽量不要使用这种方法。