golang數(shù)組和切片作為參數(shù)和返回值的實(shí)現(xiàn)
1. 數(shù)組作為參數(shù)和返回值時(shí)
1.1數(shù)組的定義
數(shù)組是具有相同唯一類(lèi)型的一組已編號(hào)且長(zhǎng)度固定的數(shù)據(jù)項(xiàng)序列,這種類(lèi)型可以是任意的原始類(lèi)型例如整型、字符串或者自定義類(lèi)型
var arr [10]int //定義長(zhǎng)度為10的類(lèi)型是int的數(shù)組arr
arr[0] = 1 // 數(shù)組的下標(biāo)從0開(kāi)始 數(shù)組的賦值
var arr1 = [5]int{1,2,3,4,5} //數(shù)組的初始化的定義方式
arr2 := [5]int{1,2,3,4,5} //同上
arr3 := [...]int{1,2,3,4,5} //編譯器通過(guò)元素個(gè)數(shù)自動(dòng)推斷數(shù)組的長(zhǎng)度1.2數(shù)組作為參數(shù)和返回值的時(shí)候
func GetArray(arr [5]int)(v [5]int){
?? ?fmt.Printf("GetArray arr===%v, %p\n", arr,&arr) // 查看傳進(jìn)來(lái)的參數(shù)和地址
?? ?arr[1] = 3 //修改其中的一個(gè)值
?? ?return arr
}
func main(){
?? ?arr :=[...]int{1,2,3,4,5} ?// 定義一個(gè)長(zhǎng)度為5 類(lèi)型是int的數(shù)組
?? ?fmt.Printf("main arr=%v, %p\n", arr, &arr)
?? ?value := GetArray(arr)
?? ?fmt.Printf("main GetArray value=%v, %p, arr=%v,%p\n", value, &value,arr, &arr)
}輸出結(jié)果:
main arr=[1 2 3 4 5], 0xc000012390
GetArray arr===[1 2 3 4 5], 0xc000012420
main GetArray value=[1 3 3 4 5], 0xc0000123f0, arr=[1 2 3 4 5],0xc000012390
從上面的輸出結(jié)果來(lái)看 當(dāng)使用數(shù)組作為參數(shù)和返回值的時(shí)候,傳進(jìn)去的是值,在函數(shù)內(nèi)部對(duì)數(shù)組進(jìn)行修改并不會(huì)影響原數(shù)據(jù)
2.切片作為參數(shù)和返回值
2.1 切片的定義初始化
Go 語(yǔ)言切片是對(duì)數(shù)組的抽象。
Go 數(shù)組的長(zhǎng)度不可改變,在特定場(chǎng)景中這樣的集合就不太適用,Go 中提供了一種靈活,功能強(qiáng)悍的內(nèi)置類(lèi)型切片(“動(dòng)態(tài)數(shù)組”),與數(shù)組相比切片的長(zhǎng)度是不固定的,可以追加元素,在追加時(shí)可能使切片的容量增大。
var myslice []int ?//定義一個(gè)切片 所有的數(shù)據(jù)類(lèi)型都是int 切片與數(shù)組的定義差別是可以不帶長(zhǎng)度
var myslice []int = make([]int ,5) //定義一個(gè)長(zhǎng)度為5的切片?
myslice1 := make([]int,5) //同上?
make([]T, length, capacity) //定義的參數(shù)分別是類(lèi)型,長(zhǎng)度,容量(可選)
myslice2 :=[] int {1,2,3 } ? //切片初始化
myarr := [...]int{1,2,3,4,5}?
myslice3 := myarr[1:3] //也可以先定義一個(gè)數(shù)組 然后截取這里是從下標(biāo)1開(kāi)始到下標(biāo)為3(不包含)ps:當(dāng)我們通過(guò)從數(shù)組截取獲得切片的時(shí)候,我們可以發(fā)現(xiàn)截取后的切片和原來(lái)的數(shù)組是共用數(shù)據(jù)源的,如果修改原來(lái)的數(shù)組數(shù)據(jù)源 那么通過(guò)截取數(shù)組得到的切片的值也會(huì)被修改反之亦然
func main(){
?? ?arr :=[...]int{1,2,3,4,5} ?// 定義一個(gè)長(zhǎng)度為5 類(lèi)型是int的數(shù)組
?? ?myslice1 := arr[0:3] //切片獲取下標(biāo)[0,3)的值
?? ?myslice2 := arr[0:4] //切片獲取下標(biāo)[1,4)的值
?? ?fmt.Printf("myslice1=%v, ppp=%p\n", myslice1, &myslice1)
?? ?fmt.Printf("myslice2=%v, ppp=%p\n", myslice2, &myslice2)
?? ?arr[1] = 666 //修改數(shù)組的值
?? ?fmt.Printf("After myslice1=%v, ppp=%p\n", myslice1, &myslice1)
?? ?fmt.Printf("After myslice2=%v, ppp=%p\n", myslice2, &myslice2)
?? ?myslice2[2] =777 //修改切片的值
?? ?fmt.Printf("arr=%v", arr)
}輸出的結(jié)果:
myslice1=[1 2 3], ppp=0xc000004078
myslice2=[1 2 3 4], ppp=0xc000004090
After myslice1=[1 666 3], ppp=0xc000004078
After myslice2=[1 666 3 4], ppp=0xc000004090
arr=[1 666 777 4 5]
2.2 切片的存儲(chǔ)大致分為3部分
一部分是存的指向匿名數(shù)組的指針,一個(gè)是長(zhǎng)度,一個(gè)是容量,我們?cè)诙x切片的時(shí)候 會(huì)在底層保存一個(gè)匿名的數(shù)組,通過(guò)上面的數(shù)組得到切片的方式得出的結(jié)論 當(dāng)我們通過(guò)一個(gè)切片得到另一個(gè)切片的時(shí)候我們的數(shù)據(jù)源也是共享的
2.3 切片作為參數(shù)和返回值
func GetSlice(myslice []int)(value []int){
?? ?fmt.Printf("GetSlice myslice===%v, ppppp=%p\n", myslice,&myslice) // 查看傳進(jìn)來(lái)的參數(shù)和地址
?? ?myslice[2] = 10
?? ?return myslice
}
func main(){
?? ?var myslice []int //定義一個(gè)空的切片
?? ?myslice = append(myslice, 1)
?? ?myslice = append(myslice, 2)
?? ?myslice = append(myslice, 3)
?? ?myslice = append(myslice, 4)
?? ?myslice = append(myslice, 5) //向切片里面追加元素
?? ?fmt.Printf("myslice ==%v ppp=%p\n", myslice, &myslice)
?? ?rslice := GetSlice(myslice)
?? ?fmt.Printf("rslice==%v ppp=%p, myslice==%v,ppp=%p\n", rslice,&rslice,myslice,&myslice)
}輸出結(jié)果:
myslice ==[1 2 3 4 5] ppp=0xc000004078
GetSlice myslice===[1 2 3 4 5], ppppp=0xc0000040c0
rslice==[1 2 10 4 5] ppp=0xc0000040a8, myslice==[1 2 10 4 5],ppp=0xc000004078
結(jié)論:從上面的輸出的結(jié)果和地址來(lái)看,當(dāng)切片作為參數(shù)的時(shí)候穿進(jìn)去的是值,也就是值傳遞,但是當(dāng)我在函數(shù)里面修改切片的時(shí)候,我們發(fā)現(xiàn)源數(shù)據(jù)也會(huì)被修改,這是因?yàn)槲覀冊(cè)谇衅牡讓泳S護(hù)這一個(gè)匿名的數(shù)組,當(dāng)我們把切片當(dāng)成參數(shù)的時(shí)候,會(huì)重現(xiàn)創(chuàng)建一個(gè)切片,但是創(chuàng)建的這個(gè)切片和我們?cè)瓉?lái)的數(shù)據(jù)是共享數(shù)據(jù)源的,所以在函數(shù)內(nèi)被修改,源數(shù)據(jù)也會(huì)被修改
2.4 append 切片動(dòng)態(tài)增長(zhǎng)的原理
golang提供了append 函數(shù)向切片中增加元素,但是切片和數(shù)組一樣也是有長(zhǎng)度的,如果添加的元素個(gè)數(shù)剛好在長(zhǎng)度范圍內(nèi),就直接在末尾添加元素,但是如果添加的元素的個(gè)數(shù)超過(guò)了長(zhǎng)度之后,就需要對(duì)底層的數(shù)組進(jìn)行擴(kuò)容了,這個(gè)新的數(shù)組的長(zhǎng)度是原來(lái)的兩倍 ,而創(chuàng)建這個(gè)新的數(shù)組之后我們將新的數(shù)組的指針保存到切片數(shù)據(jù)中,就這樣我們實(shí)現(xiàn)了切片的動(dòng)態(tài)增長(zhǎng),而當(dāng)切片作為參數(shù)的時(shí)候,如果我們?cè)诤瘮?shù)里使用append函數(shù)增加元素,且元素的個(gè)數(shù)超過(guò)長(zhǎng)度的話(huà) 在函數(shù)中我們就會(huì)創(chuàng)建除一個(gè)新的切片這個(gè)時(shí)候我們?cè)诤瘮?shù)內(nèi)對(duì)新的切片進(jìn)行修改 就不會(huì)影響到原來(lái)的切片了
func GetSlice(myslice []int)(value []int){
?? ?fmt.Printf("GetSlice myslice===%v, ppppp=%p\n", myslice,&myslice) // 查看傳進(jìn)來(lái)的參數(shù)和地址
?? ?myslice = append(myslice, 6)
?? ?return myslice
}
func main(){
?? ?myslice := make([]int,5) ?//定義長(zhǎng)度為5 類(lèi)型是int的切片
?? ?for i:=0; i< len(myslice);i++{
?? ??? ?myslice[i] = i+1
?? ?}
?? ?fmt.Printf("myslice ==%v ppp=%p\n", myslice, &myslice)
?? ?rslice := GetSlice(myslice)
?? ?fmt.Printf("rslice==%v ppp=%p, myslice==%v,ppp=%p\n", rslice,&rslice,myslice,&myslice)
}輸出結(jié)果:
myslice ==[1 2 3 4 5] ppp=0xc000004078
GetSlice myslice===[1 2 3 4 5], ppppp=0xc0000040c0
rslice==[1 2 3 4 5 6] ppp=0xc0000040a8, myslice==[1 2 3 4 5],ppp=0xc000004078
2.5 copy 函數(shù) 通過(guò)賦值切片可以使得兩個(gè)切片的數(shù)據(jù)不共享
func main(){
?? ?myslice := make([]int,5) ?//定義長(zhǎng)度為5 類(lèi)型是int的切片
?? ?for i:=0; i< len(myslice);i++{
?? ??? ?myslice[i] = i+1
?? ?}
?? ?fmt.Printf("myslice ==%v ppp=%p\n", myslice, &myslice)
?? ?copymyslice := make([]int,5)
?? ?copy(copymyslice, myslice)
?? ?myslice[4] = 10
?? ?fmt.Printf("copymyslice ==%v myslice=%v\n", copymyslice, myslice)
}輸出結(jié)果:
myslice ==[1 2 3 4 5] ppp=0xc000004078
copymyslice ==[1 2 3 4 5] myslice=[1 2 3 4 10]
結(jié)論:使用copy函數(shù)對(duì)切片進(jìn)行賦值的時(shí)候可以避免源數(shù)據(jù)與目標(biāo)數(shù)據(jù)共享底層數(shù)組
3. 總結(jié):
數(shù)組還是切片,在函數(shù)中傳遞的時(shí)候如果沒(méi)有指定為指針傳遞的話(huà),都是值傳遞,但是切片在傳遞的過(guò)程中,有著共享底層數(shù)組的風(fēng)險(xiǎn),所以如果在函數(shù)內(nèi)部進(jìn)行了更改的時(shí)候,會(huì)修改到源數(shù)據(jù),所以我們需要根據(jù)不同的需求來(lái)處理,如果我們不希望源數(shù)據(jù)被修改話(huà)的我們可以使用copy函數(shù)復(fù)制切片后再傳入,如果希望源數(shù)據(jù)被修改的話(huà)我們應(yīng)該使用指針傳遞的方式
到此這篇關(guān)于golang數(shù)組和切片作為參數(shù)和返回值的實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)golang數(shù)組和切片作為參數(shù)和返回值內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
定位并修復(fù) Go 中的內(nèi)存泄露問(wèn)題
Go 是一門(mén)帶 GC 的語(yǔ)言,這篇文章回顧了我如何發(fā)現(xiàn)內(nèi)存泄漏、如何修復(fù)它,以及我如何修復(fù) Google 示例 Go 代碼中的類(lèi)似問(wèn)題,以及我們?nèi)绾胃倪M(jìn)我們的庫(kù)以防止將來(lái)發(fā)生這種情況,感興趣的朋友一起看看吧2021-10-10
淺析Go語(yǔ)言bitset的實(shí)現(xiàn)原理
bitset包是一個(gè)將非負(fù)整數(shù)映射到布爾值的位的集合,這篇文章主要通過(guò)開(kāi)源包bitset來(lái)為大家分析一下位集合的設(shè)計(jì)和實(shí)現(xiàn),感興趣的可以學(xué)習(xí)一下2023-08-08
golang中的string與其他格式數(shù)據(jù)的轉(zhuǎn)換方法詳解
這篇文章主要介紹了golang中的string與其他格式數(shù)據(jù)的轉(zhuǎn)換方法,文章通過(guò)代碼示例介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作有一定的幫助,需要的朋友可以參考下2023-10-10
淺析Golang中調(diào)度器的關(guān)鍵機(jī)制與性能
Golang的調(diào)度器是其并發(fā)模型的核心組件,負(fù)責(zé)管理Goroutine的調(diào)度和執(zhí)行,本文將從理論和代碼層面分析Golang調(diào)度器的關(guān)鍵機(jī)制,感興趣的可以了解下2025-03-03
在Go語(yǔ)言單元測(cè)試中解決HTTP網(wǎng)絡(luò)依賴(lài)問(wèn)題
在 Go 語(yǔ)言中,我們需要找到一種可靠的方法來(lái)測(cè)試 HTTP 請(qǐng)求和響應(yīng),本文將探討在 Go 中進(jìn)行 HTTP 應(yīng)用測(cè)試時(shí),如何解決應(yīng)用程序的依賴(lài)問(wèn)題,以確保我們能夠編寫(xiě)出可靠的測(cè)試用例,需要的朋友可以參考下2023-07-07
Go語(yǔ)言時(shí)間相關(guān)操作合集(超詳細(xì))
在開(kāi)發(fā)應(yīng)用程序的過(guò)程中,經(jīng)常需要記錄某些操作的時(shí)間或者格式化時(shí)間戳,因此大部分編程語(yǔ)言都會(huì)有操作時(shí)間的庫(kù),Go語(yǔ)言當(dāng)然也不例外,本文我們就一起來(lái)學(xué)習(xí)一下time包的使用2023-08-08
GoLang OS包以及File類(lèi)型詳細(xì)講解
go中對(duì)文件和目錄的操作主要集中在os包中,下面對(duì)go中用到的對(duì)文件和目錄的操作,做一個(gè)總結(jié)筆記。在go中的文件和目錄涉及到兩種類(lèi)型,一個(gè)是type File struct,另一個(gè)是type Fileinfo interface2023-03-03

