從Java1.5開始就增加了可變參數(shù)(varargs)方法尼摹,又稱作variable arity method⊙唬可變參數(shù)方法接受0個(gè)或多個(gè)指定類型的參數(shù)悠夯。它的機(jī)制是先創(chuàng)建一個(gè)數(shù)組,數(shù)組的大小為調(diào)用位置所傳遞的參數(shù)數(shù)量竟稳,然后將值傳到數(shù)組中属桦,最后將數(shù)組傳遞到方法。
例如下面有個(gè)例子他爸,返回多個(gè)參數(shù)的和:
static int sum(int...?args)?{
int sum?=0;
for(int arg?:?args)
sum?+=?arg;
return sum;
}
很多時(shí)候聂宾,我們需要至少一個(gè)參數(shù),那么很容易想到在方法開始的時(shí)候做參數(shù)檢查诊笤,下面是一個(gè)計(jì)算參數(shù)最小值的例子:
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;
}
以上是在方法開始的時(shí)候檢查參數(shù)長(zhǎng)度是否為0系谐。但是,這是解決方案有兩個(gè)不足:1.如果沒有傳入?yún)?shù),只有在運(yùn)行的時(shí)候失敗纪他,而不是編譯的時(shí)候失敱擅骸;2.代碼不美觀茶袒,除了需要在最開始檢查有效性之外梯刚,在這個(gè)案例中,比較參數(shù)的大小的時(shí)候薪寓,只能從數(shù)組第二個(gè)開始比較亡资,代碼不夠簡(jiǎn)潔美觀。
很巧的是预愤,利用可變參數(shù)的語(yǔ)法沟于,正好有一種巧妙的方法可以解決這個(gè)問題:聲明該方法有兩個(gè)參數(shù),一個(gè)是指定類型的正常參數(shù)植康,另一個(gè)是這種類型的varargs參數(shù)旷太。這個(gè)方法彌補(bǔ)了上面的不足(不需要再檢查參數(shù)的數(shù)量了,因?yàn)橹辽僖獋鬟f一個(gè)參數(shù)销睁,否則不能通過編譯):
static int min(int firstArg , int...?remainingArgs)?{
intmin?=?firstArg;
for(intarg?:?remainingArgs)
if(arg?<?min)
min?=?arg;
return min;
}
事實(shí)上供璧,當(dāng)你真的需要讓一個(gè)方法帶有不定數(shù)量的參數(shù)的時(shí)候,可變參數(shù)才會(huì)變得非常有效冻记。它本來是為printf 和反射機(jī)制(見53條)設(shè)定的睡毒。
接下來讓我們一起看看一個(gè)有趣的例子:
List?homophones?=?Arrays.asList("to","too","two");
System.out.println(homophones);
int[]?digits?=?{1,2,3,4,5};
System.out.println(Arrays.asList(digits));
輸出結(jié)果是:
[to,?too,?two]
[[I@15db9742]
在以上的這個(gè)例子中,System.out.println調(diào)用的是toString冗栗,而List是從Object繼承了它們的toString實(shí)現(xiàn)演顾。如果使用asList方法來初始化int數(shù)組,它會(huì)忠實(shí)的將int數(shù)組包裝到List實(shí)例中隅居,打印這個(gè)List會(huì)導(dǎo)致到List中調(diào)用toString钠至,toString的是int[],打印的是數(shù)組地址胎源,得到我們并不想看到的結(jié)果棉钧。
List?list=Arrays.asList(digits);
我將代碼稍作修改,更能說明這個(gè)問題:
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);
輸出結(jié)果是:
[to,?too,?two]
[[I@15db9742]
[Ljava.lang.String;@6d06d69c]
[I@15db9742]
使用Arrays.toString(digits);方法就可以避免這個(gè)問題,專門將任何類型的數(shù)組轉(zhuǎn)換成字符串而設(shè)計(jì)
另外涕蚤,需要注意的是宪卿,在重視性能的情況下,使用可變參數(shù)機(jī)制要特別小心万栅∮蛹兀可變參數(shù)方法每次調(diào)用都會(huì)導(dǎo)致進(jìn)行一次數(shù)組分配和初始化。如果只是憑經(jīng)驗(yàn)確定烦粒,無法承受這一成本次绘,但是又需要可變參數(shù)的靈活性。這時(shí)候,需要評(píng)估邮偎,假如某個(gè)方法95%會(huì)調(diào)用3個(gè)或更少的參數(shù)管跺,那么就聲明該方法的5個(gè)重載(和上一條一樣,這幾個(gè)重載的方法必須盡量保證方法的功能相同禾进,返回值相同)豁跑,每個(gè)重載方法帶有0-3個(gè)參數(shù),超過3個(gè)參數(shù)的時(shí)候泻云,就會(huì)自動(dòng)調(diào)用可變參數(shù)方法艇拍。
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){}
像大多數(shù)的性能優(yōu)化方法一樣,這種方式看起來很不恰當(dāng)宠纯,但是用到的時(shí)候會(huì)有很大幫助卸夕。
總之,和其他規(guī)則一樣婆瓜,盡管可變參數(shù)是一個(gè)很方便的方式快集,但是它們不應(yīng)該被過度濫用。除非有必要廉白,盡量不要使用這種方法个初。