第42條 慎用可變參數

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

像大多數的性能優化方法一樣,這種方式看起來很不恰當,但是用到的時候會有很大幫助。

總之,和其他規則一樣,盡管可變參數是一個很方便的方式,但是它們不應該被過度濫用。除非有必要,盡量不要使用這種方法。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 230,622評論 6 544
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 99,716評論 3 429
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 178,746評論 0 383
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 63,991評論 1 318
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 72,706評論 6 413
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 56,036評論 1 329
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 44,029評論 3 450
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 43,203評論 0 290
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 49,725評論 1 336
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 41,451評論 3 361
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 43,677評論 1 374
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 39,161評論 5 365
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,857評論 3 351
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 35,266評論 0 28
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 36,606評論 1 295
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 52,407評論 3 400
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 48,643評論 2 380

推薦閱讀更多精彩內容