Go slice切片使用示例詳解
定義
切片區(qū)別于數(shù)組,是引用類型, 不是值類型。數(shù)組是固定長(zhǎng)度的,而切片長(zhǎng)度是可變的,我的理解是:切片是對(duì)數(shù)組一個(gè)片段的引用。
var s1 []int //定義一個(gè)存放int類型元素的切片 var s2 []string //定義一個(gè)存放string類型元素的切片 fmt.Println(s1, s2) fmt.Println(s1 == nil) //true 為空 沒(méi)有開(kāi)辟內(nèi)存空間 fmt.Println(s2 == nil) //true
打印結(jié)果:
解析: 說(shuō)明我們已經(jīng)聲明定義成功了,但是并沒(méi)有開(kāi)辟內(nèi)存空間,因?yàn)閟1、s2的值為nil
定義并初始化
我們可以在定義的同時(shí)初始化
var s1 = []int{1, 2, 3} var s2 = []string{"北苑", "長(zhǎng)陽(yáng)", "望京"} fmt.Println(s1, s2) fmt.Println(s1 == nil) //false fmt.Println(s2 == nil) //false
打印結(jié)果:
解析: 初始化成功,s1 s2的值都不等于nil
長(zhǎng)度和容量
分別使用len()、cap()獲得切片的長(zhǎng)度和容量
fmt.Printf("len(s1):%d cap(s1):%d\n", len(s1), cap(s1)) fmt.Printf("len(s2):%d cap(s2):%d\n", len(s2), cap(s2))
打印結(jié)果:
解析: 和我們預(yù)期的一致,長(zhǎng)度和容量都為3
由數(shù)組得到切片
開(kāi)篇我已經(jīng)提到數(shù)組和切片的關(guān)系,這里在進(jìn)一步講一下:
- 切片的本質(zhì)是操作數(shù)組,只是數(shù)組是固定長(zhǎng)度的,而切片的長(zhǎng)度可變的
- 切片是引用類型,可以理解為引用數(shù)組的一個(gè)片段;而數(shù)組是值類型,把數(shù)組A賦值給數(shù)組B,會(huì)為數(shù)組B開(kāi)辟新的內(nèi)存空間,修改數(shù)組B的值并不會(huì)影響數(shù)組A。而切片作為引用類型,指向同一個(gè)內(nèi)存地址,是會(huì)互相影響的。
//定義一個(gè)數(shù)組 a1 := [...]int{1, 2, 3, 4, 5, 6, 7, 8, 9} s3 := a1[0:4] //基于一個(gè)數(shù)組切割 [0:4]左包含 右不包含 即為[1,2,3,4] fmt.Println(s3)
打印結(jié)果:
注意:a1[0:4] 基于一個(gè)數(shù)組切割 [0:4]左包含 右不包含 即為[1,2,3,4]
更多切割方式舉例
a1 := [...]int{1, 2, 3, 4, 5, 6, 7, 8, 9} s4 := a1[2:4] //[3 4] s5 := a1[:4] //[1 2 3 4] s6 := a1[2:] //[3 4 5 6 7 8 9] s7 := a1[:] //[1 2 3 4 5 6 7 8 9] fmt.Println(s4) fmt.Println(s5) fmt.Println(s6) fmt.Println(s7)
打印結(jié)果:
解析: 都符合上面提到的左包含,右不包含
原則 s4從下標(biāo)2開(kāi)始截取,截取到下標(biāo)4 s5省略了第一個(gè)參數(shù),表示從下標(biāo)0開(kāi)始截取 s6省略了第二個(gè)參數(shù),表示截取到最后一個(gè)元素 s7省略了兩個(gè)參數(shù),只填寫(xiě)了中間的冒號(hào):,表示取全部元素
切片的長(zhǎng)度和容量
切片的長(zhǎng)度很好理解,就是元素的個(gè)數(shù)
切片的容量我們重點(diǎn)理解一下:在切片引用的底層數(shù)組中從切片的第一個(gè)元素到數(shù)組最后一個(gè)元素的長(zhǎng)度(元素?cái)?shù)量)
這么讀起來(lái)可能有點(diǎn)抽象,我們看下面這個(gè)栗子就很好理解啦:
a1 := [...]int{1, 2, 3, 4, 5, 6, 7, 8, 9} s5 := a1[:4] //[1 2 3 4] s6 := a1[2:] //[3 4 5 6 7 8 9] s7 := a1[:] //[1 2 3 4 5 6 7 8 9] fmt.Printf("len(s5):%d cap(s5):%d\n", len(s5), cap(s5)) //4 9 fmt.Printf("len(s6):%d cap(s6):%d\n", len(s6), cap(s6)) //7 7 fmt.Printf("len(s7):%d cap(s7):%d\n", len(s7), cap(s7)) //9 9
打印結(jié)果:
解析: a1是數(shù)組長(zhǎng)度為9,容量也為9,值是從1~9
s5/s6/s7都是切割數(shù)組a1得到的切片。
s5的長(zhǎng)度為4,因?yàn)橹挥? 2 3 4這4個(gè)元素,容量為9,因?yàn)閟5切片的第一個(gè)元素是1,而s5底層數(shù)組a1最后一個(gè)元素是9,1~9共9個(gè)元素,所以s5的容量為9。
s6的長(zhǎng)度為7,因?yàn)閟6的元素是39這7個(gè)元素;容量也為7,因?yàn)閟5的底層數(shù)組最后一個(gè)元素是9,39共7個(gè)元素,所以s6的容量為7。
S7更好理解了,長(zhǎng)度和容量都是9,小伙伴們自己理解一下。
切片再切片
我們可以對(duì)切片進(jìn)行再切片操作
比如,我們針對(duì)上面的數(shù)據(jù)再次切片進(jìn)行測(cè)試
s8 :=s6[3:] fmt.Printf("len(s8):%d cap(s8):%d\n", len(s8), cap(s8)) //4 4
打印結(jié)果:
解析:我們知道可以對(duì)切片進(jìn)行再次切片就可以,至于長(zhǎng)度和容器大家搞明白上面的栗子,這個(gè)輸出結(jié)果就是意料之中的了。
slice是引用類型
我們舉個(gè)栗子來(lái)證明切片是引用類型
//定義數(shù)組 a1 := [...]int{1, 2, 3, 4, 5, 6, 7, 8, 9} //有數(shù)組切割成切片s6 s6 := a1[2:] //[3 4 5 6 7 8 9] //切片再次切片,賦值給s8 s8 :=s6[3:] //[6 7 8 9] //修改原始數(shù)組,把下標(biāo)為2的值由3改為333 a1[2] = 333 //打印s6,發(fā)現(xiàn)s6中的3也變成了333 fmt.Println("s6:", s6) //[333 4 5 6 7 8 9] //因?yàn)閟8基于s6切片而成,我們測(cè)試一下切片再切片的引用傳的 fmt.Println("s8:", s8) //[6 7 8 9] //我們把原始數(shù)組下標(biāo)為5的值由6改為666 a1[5] = 666 //打印s8切片,得到結(jié)果6也變成了666 fmt.Println("s8:", s8) //[666 7 8 9]
打印結(jié)果:
解析: 由此我們可以明確的知道切片是引用類型,當(dāng)?shù)讓訑?shù)組改變時(shí),不管是切片,還是切片再切片,值都會(huì)改變。因?yàn)樗麄兪褂玫氖且粋€(gè)內(nèi)存塊,引用的一個(gè)內(nèi)存地址。
總結(jié)
這篇文章介紹了切片的特點(diǎn),如何定義切片,如果由數(shù)組切割切片,切片的引用類型特征,更多關(guān)于Go slice切片的資料請(qǐng)關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
golang分層測(cè)試之http接口測(cè)試入門(mén)教程
這篇文章主要介紹了golang分層測(cè)試之http接口測(cè)試入門(mén)教程,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧2018-12-12深入理解Golang?make和new的區(qū)別及實(shí)現(xiàn)原理
在Go語(yǔ)言中,有兩個(gè)比較雷同的內(nèi)置函數(shù),分別是new和make方法,二者都可以用來(lái)分配內(nèi)存,那他們有什么區(qū)別呢?下面我們就從底層來(lái)分析一下二者的不同。感興趣的小伙伴們可以參考借鑒,希望對(duì)大家能有所幫助2022-10-10Go中的 panic / recover 簡(jiǎn)介與實(shí)踐記錄
這篇文章主要介紹了Go中的 panic / recover 簡(jiǎn)介與實(shí)踐,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2023-04-04使用Golong實(shí)現(xiàn)JWT身份驗(yàn)證的詳細(xì)過(guò)程
JWT提供了一種強(qiáng)大而靈活的方法來(lái)處理Web應(yīng)用程序中的身份驗(yàn)證和授權(quán),本教程將引導(dǎo)您逐步實(shí)現(xiàn)Go應(yīng)用程序中的JWT身份驗(yàn)證過(guò)程,感興趣的朋友跟隨小編一起看看吧2024-03-03夯實(shí)Golang基礎(chǔ)之?dāng)?shù)據(jù)類型梳理匯總
這篇文章主要8為大家介紹了夯實(shí)Golang基礎(chǔ)之?dāng)?shù)據(jù)類型梳理匯總,有需要的朋友可以借鑒參考下,希望能夠有所幫助2023-10-10golang解析json數(shù)據(jù)的4種方法總結(jié)
在日常工作中每一名開(kāi)發(fā)者,不管是前端還是后端,都經(jīng)常使用 JSON,下面這篇文章主要給大家介紹了關(guān)于golang解析json數(shù)據(jù)的4種方法,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2023-06-06golang框架gin的日志處理和zap lumberjack日志使用方式
這篇文章主要介紹了golang框架gin的日志處理和zap lumberjack日志使用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-01-01