一文帶你深入了解Go語(yǔ)言中切片的奧秘
Go語(yǔ)言基礎(chǔ)三
切片的定義
1. 切片:切片是數(shù)組的一個(gè)引用,因此切片是引用類型。但自身是結(jié)構(gòu)體,值拷貝傳遞。
2. 切片的長(zhǎng)度可以改變,因此,切片是一個(gè)可變的數(shù)組。
3. 切片遍歷方式和數(shù)組一樣,可以用len()求長(zhǎng)度。表示可用元素?cái)?shù)量,讀寫(xiě)操作不能超過(guò)該限制。
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)
}我們首先定義了一個(gè)未進(jìn)行初始化變量,名字為s1,數(shù)據(jù)類型為int類型的數(shù)組,同時(shí)運(yùn)用if進(jìn)行判斷,由本題的demo可知,該答案為空;接下來(lái)對(duì)s1、s2、s3這三個(gè)數(shù)組進(jìn)行打印輸出,由于三個(gè)數(shù)組均未進(jìn)行初始化操作,因此三個(gè)數(shù)組打印出來(lái)的值都是[];s4數(shù)組也未進(jìn)行初始化的操作,因此s4數(shù)組打印輸出也是[];s5數(shù)組顧名思義,45 打印出來(lái)即為[1,2,3];我們對(duì)s6數(shù)組進(jìn)行初始化操作,并且用arr數(shù)組的元素進(jìn)行處理,使用切片的方法,對(duì)s6進(jìn)行切片,最終打印出來(lái)的結(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] //可以簡(jiǎn)寫(xiě)為 var slice []int = arr[:end]
var slice2 []int = arr[5:10] //可以簡(jiǎn)寫(xiě)為 var slice[]int = arr[start:]
var slice3 []int = arr[0:len(arr)] //var slice []int = arr[:]
var slice4 = arr[:len(arr)-1] //去掉切片的最后一個(gè)元素
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] //可以簡(jiǎn)寫(xiě)為 slice := arr[:end]
slice7 := arr[5:10] //可以簡(jiǎn)寫(xiě)為 slice := arr[start:]
slice8 := arr[0:len(arr)] //slice := arr[:]
slice9 := arr[:len(arr)-1] //去掉切片的最后一個(gè)元素
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ù)代碼要求定義四個(gè)數(shù)組,類型均為int類型,并對(duì)它們分別進(jìn)行切片處理。我們?cè)谔幚?strong>slice4數(shù)組的時(shí)候,做法是去掉切片的最后一個(gè)元素
上述運(yùn)算結(jié)果顯而易見(jiàn),作者在這里不再進(jìn)行闡述,如有不會(huì)的,可詳細(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ù)中,我們定義了一個(gè)名為data,長(zhǎng)度為任意類型,數(shù)據(jù)類型為int類型的數(shù)組,并進(jìn)行初始化賦值的操作;使用變量s對(duì)數(shù)組data進(jìn)行切片處理,同時(shí)秉持著包前不包后的語(yǔ)法規(guī)則,顧切片元素應(yīng)為{2,3},在下一步的賦值操作里面,我們將2進(jìn)行賦值操作(索引為0),得到的結(jié)果為102 ;同理,s[1]得到的值為203。緊接著最后一個(gè)打印輸出大家也很清楚啦,就不作贅述了
package main
?
import "fmt"
?
func main() {
s1 := []int{0, 1, 2, 3, 8: 100} // 通過(guò)初始化表達(dá)式構(gòu)造,可使用索引號(hào)。
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))
}由上方代碼可知,我們定義了一個(gè)名為s1,數(shù)組長(zhǎng)度為9,數(shù)據(jù)類型為int類型的數(shù)組。我們?cè)谡{(diào)用函數(shù)打印輸出的時(shí)候,由于數(shù)組中第9個(gè)元素被替換為100,所以最終的結(jié)果是[0 1 2 3 0 0 0 0 100] 9 9,后兩個(gè)數(shù)組同理;值得一提的是,如果我們省略寫(xiě)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)
}我們定義了一個(gè)數(shù)組,里面存放了4個(gè)元素,接著我們使用&來(lái)獲取底層數(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])
?
}我們?cè)?strong>main方法里面定義了一個(gè)函數(shù)體結(jié)構(gòu),在函數(shù)體里面定義了一個(gè)int類型的變量,名為x,緊接著我們將數(shù)組d拷貝到s中,同時(shí)將d數(shù)組中索引為1的元素賦值為10;同理,索引為2的元素賦值為20,因此打印的結(jié)果為[{0} {10} {20} {0} {0}],緊接著關(guān)于地址值的打印就不做贅述啦
到此這篇關(guān)于一文帶你深入了解Go語(yǔ)言中切片的奧秘的文章就介紹到這了,更多相關(guān)Go語(yǔ)言切片內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
Golang 使用map需要注意的幾個(gè)點(diǎn)
這篇文章主要介紹了Golang 使用map需要注意的幾個(gè)點(diǎn),幫助大家更好的理解和學(xué)習(xí)golang,感興趣的朋友可以了解下2020-09-09
Goland遠(yuǎn)程連接Linux進(jìn)行項(xiàng)目開(kāi)發(fā)的實(shí)現(xiàn)
有的時(shí)候我們的開(kāi)發(fā)代碼要在linux服務(wù)器上運(yùn)行,本文主要介紹了Goland遠(yuǎn)程連接Linux進(jìn)行項(xiàng)目開(kāi)發(fā)的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下2024-06-06
Go Asynq異步任務(wù)處理的實(shí)現(xiàn)
Asynq是一個(gè)新興的異步任務(wù)處理解決方案,它提供了輕量級(jí)的、易于使用的API,本文主要介紹了Go Asynq異步任務(wù)處理的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的可以了解一下2023-06-06
golang讀取http的body時(shí)遇到的坑及解決
這篇文章主要介紹了golang讀取http的body時(shí)遇到的坑及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2024-03-03
Go語(yǔ)言中一定要知道的切片使用注意事項(xiàng)總結(jié)
了解和掌握切片的使用注意事項(xiàng),可以避免意外的程序行為,所以本文就來(lái)和大家深入探討一下Go語(yǔ)言切片常見(jiàn)的注意事項(xiàng),希望對(duì)大家有所幫助2023-06-06

