一文帶你深入了解Go語言中切片的奧秘
Go語言基礎(chǔ)三
切片的定義
1. 切片:切片是數(shù)組的一個引用,因此切片是引用類型。但自身是結(jié)構(gòu)體,值拷貝傳遞。
2. 切片的長度可以改變,因此,切片是一個可變的數(shù)組。
3. 切片遍歷方式和數(shù)組一樣,可以用len()求長度。表示可用元素?cái)?shù)量,讀寫操作不能超過該限制。
4. cap可以求出slice最大擴(kuò)張容量,不能超出數(shù)組限制。0 <= len(slice) <= len(array),其中array是slice引用的數(shù)組。
5. 切片的定義:var 變量名 []類型,比如 var str []string var arr []int。
6. 如果 slice == nil,那么 len、cap 結(jié)果都等于 0。
創(chuàng)建切片的方式
package main ? import "fmt" func main() { var s1 []int if s1 == nil { fmt.Println("是空") } else { fmt.Println("不是空") } s2 := []int{} var s3 []int = make([]int, 0) fmt.Println(s1, s2, s3) var s4 []int = make([]int, 0, 0) fmt.Println(s4) s5 := []int{1, 2, 3} fmt.Println(s5) arr := [5]int{1, 2, 3, 4, 5} var s6 []int s6 = arr[1:4] fmt.Println(s6) }
我們首先定義了一個未進(jìn)行初始化變量,名字為s1,數(shù)據(jù)類型為int類型的數(shù)組,同時運(yùn)用if進(jìn)行判斷,由本題的demo可知,該答案為空;接下來對s1、s2、s3這三個數(shù)組進(jìn)行打印輸出,由于三個數(shù)組均未進(jìn)行初始化操作,因此三個數(shù)組打印出來的值都是[]
;s4數(shù)組也未進(jìn)行初始化的操作,因此s4數(shù)組打印輸出也是[]
;s5數(shù)組顧名思義,45 打印出來即為[1,2,3]
;我們對s6數(shù)組進(jìn)行初始化操作,并且用arr數(shù)組的元素進(jìn)行處理,使用切片的方法,對s6進(jìn)行切片,最終打印出來的結(jié)果為[2,3,4]
切片初始化
package main ? import ( "fmt" ) ? var arr = [...]int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9} var slice0 []int = arr[2:8] var slice1 []int = arr[0:6] //可以簡寫為 var slice []int = arr[:end] var slice2 []int = arr[5:10] //可以簡寫為 var slice[]int = arr[start:] var slice3 []int = arr[0:len(arr)] //var slice []int = arr[:] var slice4 = arr[:len(arr)-1] //去掉切片的最后一個元素 func main() { fmt.Printf("全局變量:arr %v\n", arr) fmt.Printf("全局變量:slice0 %v\n", slice0) fmt.Printf("全局變量:slice1 %v\n", slice1) fmt.Printf("全局變量:slice2 %v\n", slice2) fmt.Printf("全局變量:slice3 %v\n", slice3) fmt.Printf("全局變量:slice4 %v\n", slice4) fmt.Printf("-----------------------------------\n") arr2 := [...]int{9, 8, 7, 6, 5, 4, 3, 2, 1, 0} slice5 := arr[2:8] slice6 := arr[0:6] //可以簡寫為 slice := arr[:end] slice7 := arr[5:10] //可以簡寫為 slice := arr[start:] slice8 := arr[0:len(arr)] //slice := arr[:] slice9 := arr[:len(arr)-1] //去掉切片的最后一個元素 fmt.Printf("局部變量: arr2 %v\n", arr2) fmt.Printf("局部變量: slice5 %v\n", slice5) fmt.Printf("局部變量: slice6 %v\n", slice6) fmt.Printf("局部變量: slice7 %v\n", slice7) fmt.Printf("局部變量: slice8 %v\n", slice8) fmt.Printf("局部變量: slice9 %v\n", slice9) }
首先我們根據(jù)代碼要求定義四個數(shù)組,類型均為int類型,并對它們分別進(jìn)行切片處理。我們在處理slice4數(shù)組的時候,做法是去掉切片的最后一個元素
上述運(yùn)算結(jié)果顯而易見,作者在這里不再進(jìn)行闡述,如有不會的,可詳細(xì)看數(shù)組的那一篇文章的講解
package main ? import ( "fmt" ) ? var slice0 []int = make([]int, 10) var slice1 = make([]int, 10) var slice2 = make([]int, 10, 10) ? func main() { fmt.Printf("make全局slice0 :%v\n", slice0) fmt.Printf("make全局slice1 :%v\n", slice1) fmt.Printf("make全局slice2 :%v\n", slice2) fmt.Println("--------------------------------------") slice3 := make([]int, 10) slice4 := make([]int, 10) slice5 := make([]int, 10, 10) fmt.Printf("make局部slice3 :%v\n", slice3) fmt.Printf("make局部slice4 :%v\n", slice4) fmt.Printf("make局部slice5 :%v\n", slice5) }
由于上述的全局變量和局部變量均未進(jìn)行初始化的操作,因此數(shù)組的值全部為0
package main ? import ( "fmt" ) ? func main() { data := [...]int{0, 1, 2, 3, 4, 5} ? s := data[2:4] s[0] += 100 s[1] += 200 ? fmt.Println(s) fmt.Println(data) }
在main函數(shù)中,我們定義了一個名為data,長度為任意類型,數(shù)據(jù)類型為int類型的數(shù)組,并進(jìn)行初始化賦值的操作;使用變量s對數(shù)組data進(jìn)行切片處理,同時秉持著包前不包后的語法規(guī)則,顧切片元素應(yīng)為{2,3}
,在下一步的賦值操作里面,我們將2進(jìn)行賦值操作(索引為0),得到的結(jié)果為102 ;同理,s[1]
得到的值為203。緊接著最后一個打印輸出大家也很清楚啦,就不作贅述了
package main ? import "fmt" ? func main() { s1 := []int{0, 1, 2, 3, 8: 100} // 通過初始化表達(dá)式構(gòu)造,可使用索引號。 fmt.Println(s1, len(s1), cap(s1)) ? s2 := make([]int, 6, 8) // 使用 make 創(chuàng)建,指定 len 和 cap 值。 fmt.Println(s2, len(s2), cap(s2)) ? s3 := make([]int, 6) // 省略 cap,相當(dāng)于 cap = len。 fmt.Println(s3, len(s3), cap(s3)) }
由上方代碼可知,我們定義了一個名為s1,數(shù)組長度為9,數(shù)據(jù)類型為int類型的數(shù)組。我們在調(diào)用函數(shù)打印輸出的時候,由于數(shù)組中第9個元素被替換為100,所以最終的結(jié)果是[0 1 2 3 0 0 0 0 100] 9 9
,后兩個數(shù)組同理;值得一提的是,如果我們省略寫cap,則默認(rèn)為cap = len
package main ? import "fmt" ? func main() { s := []int{0, 1, 2, 3} p := &s[2] // *int, 獲取底層數(shù)組元素指針。 *p += 100 ? fmt.Println(s) }
我們定義了一個數(shù)組,里面存放了4個元素,接著我們使用&來獲取底層數(shù)組元素指針,然后使其帶有+100的操作,最終索引為3的元素就被賦值成了102,因此最后的結(jié)果是[0 1 102 3]
package main ? import ( "fmt" ) ? func main() { d := [5]struct { x int }{} ? s := d[:] ? d[1].x = 10 s[2].x = 20 ? fmt.Println(d) fmt.Printf("%p, %p\n", &d, &d[0]) ? }
我們在main方法里面定義了一個函數(shù)體結(jié)構(gòu),在函數(shù)體里面定義了一個int類型的變量,名為x,緊接著我們將數(shù)組d拷貝到s中,同時將d數(shù)組中索引為1的元素賦值為10;同理,索引為2的元素賦值為20,因此打印的結(jié)果為[{0} {10} {20} {0} {0}]
,緊接著關(guān)于地址值的打印就不做贅述啦
到此這篇關(guān)于一文帶你深入了解Go語言中切片的奧秘的文章就介紹到這了,更多相關(guān)Go語言切片內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
go實(shí)現(xiàn)文件的創(chuàng)建、刪除與讀取示例代碼
這篇文章主要給大家介紹了關(guān)于go如何實(shí)現(xiàn)文件的創(chuàng)建、刪除與讀取的相關(guān)資料,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面來一起看看吧2019-02-02go語言使用第三方包 json化結(jié)構(gòu)體操作示例
這篇文章主要介紹了go語言使用第三方包 json化結(jié)構(gòu)體操作,結(jié)合實(shí)例形式分析了Go語言ffjson包git安裝及結(jié)構(gòu)體轉(zhuǎn)json字符串相關(guān)操作技巧,需要的朋友可以參考下2019-06-06golang 檢查網(wǎng)絡(luò)狀態(tài)是否正常的方法
今天小編就為大家分享一篇golang 檢查網(wǎng)絡(luò)狀態(tài)是否正常的方法,具有很好的參考價(jià)值,希望對大家有所幫助。一起跟隨小編過來看看吧2019-07-07Windows下CMD執(zhí)行Go出現(xiàn)中文亂碼的解決方法
本文主要介紹了Windows下CMD執(zhí)行Go出現(xiàn)中文亂碼的解決方法,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2023-02-02Golang標(biāo)準(zhǔn)庫之errors包應(yīng)用方式
Go語言的errors包提供了基礎(chǔ)的錯誤處理能力,允許通過errors.New創(chuàng)建自定義error對象,error在Go中是一個接口,通過實(shí)現(xiàn)Error方法來定義錯誤文本,對錯誤的比較通?;趯ο蟮刂?而非文本內(nèi)容,因此即使兩個錯誤文本相同2024-10-10