Effective Java

第42条 慎用可变参数

2017-07-10  本文已影响0人  明月_48cd

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){}

像大多数的性能优化方法一样,这种方式看起来很不恰当,但是用到的时候会有很大帮助。

总之,和其他规则一样,尽管可变参数是一个很方便的方式,但是它们不应该被过度滥用。除非有必要,尽量不要使用这种方法。

上一篇下一篇

猜你喜欢

热点阅读