12. 可變參數(shù)函數(shù)

12. 可變參數(shù)函數(shù)

什么是可變參數(shù)函數(shù)

可變參數(shù)函數(shù)是一種參數(shù)個(gè)數(shù)可變的函數(shù)素标。

語(yǔ)法

如果函數(shù)最后一個(gè)參數(shù)被記作 ...T ,這時(shí)函數(shù)可以接受任意個(gè) T 類(lèi)型參數(shù)作為最后一個(gè)參數(shù)蒸痹。

請(qǐng)注意只有函數(shù)的最后一個(gè)參數(shù)才允許是可變的。

通過(guò)一些例子理解可變參數(shù)函數(shù)如何工作

你是否曾經(jīng)想過(guò) append 函數(shù)是如何將任意個(gè)參數(shù)值加入到切片中的呛哟。這樣 append 函數(shù)可以接受不同數(shù)量的參數(shù)叠荠。

func append(slice []Type, elems ...Type) []Type

上面是 append 函數(shù)的定義。在定義中 elems 是可變參數(shù)扫责。這樣 append 函數(shù)可以接受可變化的參數(shù)榛鼎。

讓我們創(chuàng)建一個(gè)我們自己的可變參數(shù)函數(shù)。我們將寫(xiě)一段簡(jiǎn)單的程序鳖孤,在輸入的整數(shù)列表里查找某個(gè)整數(shù)是否存在者娱。

package main

import (
    "fmt"
)

func find(num int, nums ...int) {
    fmt.Printf("type of nums is %T\n", nums)
    found := false
    for i, v := range nums {
        if v == num {
            fmt.Println(num, "found at index", i, "in", nums)
            found = true
        }
    }
    if !found {
        fmt.Println(num, "not found in ", nums)
    }
    fmt.Printf("\n")
}
func main() {
    find(89, 89, 90, 95)
    find(45, 56, 67, 45, 90, 109)
    find(78, 38, 56, 98)
    find(87)
}

在上面程序中 func find(num int, nums ...int) 中的 nums 可接受任意數(shù)量的參數(shù)。在 find 函數(shù)中苏揣,參數(shù) nums 相當(dāng)于一個(gè)整型切片黄鳍。

可變參數(shù)函數(shù)的工作原理是把可變參數(shù)轉(zhuǎn)換為一個(gè)新的切片。以上面程序中的第 22 行為例平匈,find 函數(shù)中的可變參數(shù)是 89框沟,90,95 吐葱。 find 函數(shù)接受一個(gè) int 類(lèi)型的可變參數(shù)街望。因此這三個(gè)參數(shù)被編譯器轉(zhuǎn)換為一個(gè) int 類(lèi)型切片 int []int{89, 90, 95} 然后被傳入 find函數(shù)。

在第 10 行弟跑, for 循環(huán)遍歷 nums 切片,如果 num 在切片中灾前,則打印 num 的位置。如果 num 不在切片中,則打印提示未找到該數(shù)字孟辑。

上面代碼的輸出值如下,

type of nums is []int
89 found at index 0 in [89 90 95]

type of nums is []int
45 found at index 2 in [56 67 45 90 109]

type of nums is []int
78 not found in  [38 56 98]

type of nums is []int
87 not found in  []

在上面程序的第 25 行哎甲,find 函數(shù)僅有一個(gè)參數(shù)。我們沒(méi)有給可變參數(shù) nums ...int 傳入任何參數(shù)饲嗽。這也是合法的炭玫,在這種情況下 nums 是一個(gè)長(zhǎng)度和容量為 0 的 nil 切片。

給可變參數(shù)函數(shù)傳入切片

下面例子中貌虾,我們給可變參數(shù)函數(shù)傳入一個(gè)切片吞加,看看會(huì)發(fā)生什么。

package main

import (
    "fmt"
)

func find(num int, nums ...int) {
    fmt.Printf("type of nums is %T\n", nums)
    found := false
    for i, v := range nums {
        if v == num {
            fmt.Println(num, "found at index", i, "in", nums)
            found = true
        }
    }
    if !found {
        fmt.Println(num, "not found in ", nums)
    }
    fmt.Printf("\n")
}
func main() {
    nums := []int{89, 90, 95}
    find(89, nums)
}

在第 23 行中,我們將一個(gè)切片傳給一個(gè)可變參數(shù)函數(shù)衔憨。

這種情況下無(wú)法通過(guò)編譯叶圃,編譯器報(bào)出錯(cuò)誤 main.go:23: cannot use nums (type []int) as type int in argument to find

為什么無(wú)法工作呢践图?原因很直接掺冠,find 函數(shù)的說(shuō)明如下,

func find(num int, nums ...int)

由可變參數(shù)函數(shù)的定義可知码党,nums ...int 意味它可以接受 int 類(lèi)型的可變參數(shù)德崭。

在上面程序的第 23 行,nums 作為可變參數(shù)傳入 find 函數(shù)揖盘。前面我們知道眉厨,這些可變參數(shù)參數(shù)會(huì)被轉(zhuǎn)換為 int 類(lèi)型切片然后在傳入 find 函數(shù)中。但是在這里 nums 已經(jīng)是一個(gè) int 類(lèi)型切片扣讼,編譯器試圖在 nums 基礎(chǔ)上再創(chuàng)建一個(gè)切片缺猛,像下面這樣

find(89, []int{nums})

這里之所以會(huì)失敗是因?yàn)?nums 是一個(gè) []int類(lèi)型 而不是 int類(lèi)型。

那么有沒(méi)有辦法給可變參數(shù)函數(shù)傳入切片參數(shù)呢椭符?答案是肯定的。

有一個(gè)可以直接將切片傳入可變參數(shù)函數(shù)的語(yǔ)法糖耻姥,你可以在在切片后加上 ... 后綴销钝。如果這樣做,切片將直接傳入函數(shù)琐簇,不再創(chuàng)建新的切片

在上面的程序中蒸健,如果你將第 23 行的 find(89, nums) 替換為 find(89, nums...) ,程序?qū)⒊晒幾g并有如下輸出

type of nums is []int
89 found at index 0 in [89 90 95]

下面是完整的程序供您參考婉商。

package main

import (
    "fmt"
)

func find(num int, nums ...int) {
    fmt.Printf("type of nums is %T\n", nums)
    found := false
    for i, v := range nums {
        if v == num {
            fmt.Println(num, "found at index", i, "in", nums)
            found = true
        }
    }
    if !found {
        fmt.Println(num, "not found in ", nums)
    }
    fmt.Printf("\n")
}
func main() {
    nums := []int{89, 90, 95}
    find(89, nums...)
}

不直觀(guān)的錯(cuò)誤

當(dāng)你修改可變參數(shù)函數(shù)中的切片時(shí)似忧,請(qǐng)確保你知道你正在做什么。

下面讓我們來(lái)看一個(gè)簡(jiǎn)單的例子丈秩。

package main

import (
    "fmt"
)

func change(s ...string) {  
    s[0] = "Go"
}

func main() {
    welcome := []string{"hello", "world"}
    change(welcome...)
    fmt.Println(welcome)
}

你認(rèn)為這段代碼將輸出什么呢盯捌?如果你認(rèn)為它輸出 [Go world] 。恭喜你蘑秽!你已經(jīng)理解了可變參數(shù)函數(shù)和切片饺著。如果你猜錯(cuò)了,那也不要緊肠牲,讓我來(lái)解釋下為什么會(huì)有這樣的輸出幼衰。

在第 13 行,我們使用了語(yǔ)法糖 ... 并且將切片作為可變參數(shù)傳入 change 函數(shù)缀雳。

正如前面我們所討論的渡嚣,如果使用了 ...welcome 切片本身會(huì)作為參數(shù)直接傳入,不需要再創(chuàng)建一個(gè)新的切片识椰。這樣參數(shù) welcome 將作為參數(shù)傳入 change 函數(shù)

change 函數(shù)中扬绪,切片的第一個(gè)元素被替換成 Go,這樣程序產(chǎn)生了下面的輸出值

[Go world]

這里還有一個(gè)例子來(lái)理解可變參數(shù)函數(shù)裤唠。

package main

import (
    "fmt"
)

func change(s ...string) {
    s[0] = "Go"
    s = append(s, "playground")
    fmt.Println(s)
}

func main() {
    welcome := []string{"hello", "world"}
    change(welcome...)
    fmt.Println(welcome)
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末挤牛,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子种蘸,更是在濱河造成了極大的恐慌墓赴,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,454評(píng)論 6 493
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件航瞭,死亡現(xiàn)場(chǎng)離奇詭異诫硕,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)刊侯,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,553評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門(mén)章办,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人滨彻,你說(shuō)我怎么就攤上這事藕届。” “怎么了亭饵?”我有些...
    開(kāi)封第一講書(shū)人閱讀 157,921評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵休偶,是天一觀(guān)的道長(zhǎng)。 經(jīng)常有香客問(wèn)我辜羊,道長(zhǎng)踏兜,這世上最難降的妖魔是什么? 我笑而不...
    開(kāi)封第一講書(shū)人閱讀 56,648評(píng)論 1 284
  • 正文 為了忘掉前任八秃,我火速辦了婚禮碱妆,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘昔驱。我一直安慰自己疹尾,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,770評(píng)論 6 386
  • 文/花漫 我一把揭開(kāi)白布舍悯。 她就那樣靜靜地躺著航棱,像睡著了一般。 火紅的嫁衣襯著肌膚如雪萌衬。 梳的紋絲不亂的頭發(fā)上饮醇,一...
    開(kāi)封第一講書(shū)人閱讀 49,950評(píng)論 1 291
  • 那天,我揣著相機(jī)與錄音秕豫,去河邊找鬼朴艰。 笑死观蓄,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的祠墅。 我是一名探鬼主播侮穿,決...
    沈念sama閱讀 39,090評(píng)論 3 410
  • 文/蒼蘭香墨 我猛地睜開(kāi)眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼毁嗦!你這毒婦竟也來(lái)了亲茅?” 一聲冷哼從身側(cè)響起,我...
    開(kāi)封第一講書(shū)人閱讀 37,817評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤狗准,失蹤者是張志新(化名)和其女友劉穎克锣,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇?shù)林里發(fā)現(xiàn)了一具尸體腔长,經(jīng)...
    沈念sama閱讀 44,275評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡袭祟,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,592評(píng)論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了捞附。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片巾乳。...
    茶點(diǎn)故事閱讀 38,724評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖鸟召,靈堂內(nèi)的尸體忽然破棺而出胆绊,到底是詐尸還是另有隱情,我是刑警寧澤药版,帶...
    沈念sama閱讀 34,409評(píng)論 4 333
  • 正文 年R本政府宣布辑舷,位于F島的核電站,受9級(jí)特大地震影響槽片,放射性物質(zhì)發(fā)生泄漏。R本人自食惡果不足惜肢础,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,052評(píng)論 3 316
  • 文/蒙蒙 一还栓、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧传轰,春花似錦剩盒、人聲如沸。這莊子的主人今日做“春日...
    開(kāi)封第一講書(shū)人閱讀 30,815評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至期贫,卻和暖如春跟匆,著一層夾襖步出監(jiān)牢的瞬間,已是汗流浹背通砍。 一陣腳步聲響...
    開(kāi)封第一講書(shū)人閱讀 32,043評(píng)論 1 266
  • 我被黑心中介騙來(lái)泰國(guó)打工玛臂, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留烤蜕,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,503評(píng)論 2 361
  • 正文 我出身青樓迹冤,卻偏偏與公主長(zhǎng)得像讽营,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子泡徙,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,627評(píng)論 2 350

推薦閱讀更多精彩內(nèi)容