一文帶你深入瞭解Go語言中切片的奧秘

Go語言基礎三

切片的定義

1. 切片:切片是數組的一個引用,因此切片是引用類型。但自身是結構體,值拷貝傳遞。

2. 切片的長度可以改變,因此,切片是一個可變的數組。

3. 切片遍歷方式和數組一樣,可以用len()求長度。表示可用元素數量,讀寫操作不能超過該限制。 

4. cap可以求出slice最大擴張容量,不能超出數組限制。0 <= len(slice) <= len(array),其中array是slice引用的數組。

5. 切片的定義:var 變量名 []類型,比如 var str []string  var arr []int。

6. 如果 slice == nil,那麼 len、cap 結果都等於 0。

創建切片的方式

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)
}

我們首先定義瞭一個未進行初始化變量,名字為s1,數據類型為int類型的數組,同時運用if進行判斷,由本題的demo可知,該答案為;接下來對s1、s2、s3這三個數組進行打印輸出,由於三個數組均未進行初始化操作,因此三個數組打印出來的值都是[]s4數組也未進行初始化的操作,因此s4數組打印輸出也是[];s5數組顧名思義,45 打印出來即為[1,2,3];我們對s6數組進行初始化操作,並且用arr數組的元素進行處理,使用切片的方法,對s6進行切片,最終打印出來的結果為[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)
}

首先我們根據代碼要求定義四個數組,類型均為int類型,並對它們分別進行切片處理。我們在處理slice4數組的時候,做法是去掉切片的最後一個元素

上述運算結果顯而易見,作者在這裡不再進行闡述,如有不會的,可詳細看數組的那一篇文章的講解

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)
}

由於上述的全局變量和局部變量均未進行初始化的操作,因此數組的值全部為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函數中,我們定義瞭一個名為data,長度為任意類型,數據類型為int類型的數組,並進行初始化賦值的操作;使用變量s對數組data進行切片處理,同時秉持著包前不包後的語法規則,顧切片元素應為{2,3},在下一步的賦值操作裡面,我們將2進行賦值操作(索引為0),得到的結果為102 ;同理,s[1]得到的值為203。緊接著最後一個打印輸出大傢也很清楚啦,就不作贅述瞭

package main
​
import "fmt"
​
func main() {
    s1 := []int{0, 1, 2, 3, 8: 100} // 通過初始化表達式構造,可使用索引號。
    fmt.Println(s1, len(s1), cap(s1))
​
    s2 := make([]int, 6, 8) // 使用 make 創建,指定 len 和 cap 值。
    fmt.Println(s2, len(s2), cap(s2))
​
    s3 := make([]int, 6) // 省略 cap,相當於 cap = len。
    fmt.Println(s3, len(s3), cap(s3))
}

由上方代碼可知,我們定義瞭一個名為s1,數組長度為9,數據類型為int類型的數組。我們在調用函數打印輸出的時候,由於數組中第9個元素被替換為100,所以最終的結果是[0 1 2 3 0 0 0 0 100] 9 9,後兩個數組同理;值得一提的是,如果我們省略寫cap,則默認為cap = len

package main
​
import "fmt"
​
func main() {
    s := []int{0, 1, 2, 3}
    p := &s[2] // *int, 獲取底層數組元素指針。
    *p += 100
​
    fmt.Println(s)
}

我們定義瞭一個數組,裡面存放瞭4個元素,接著我們使用&來獲取底層數組元素指針,然後使其帶有+100的操作,最終索引為3的元素就被賦值成瞭102,因此最後的結果是[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方法裡面定義瞭一個函數體結構,在函數體裡面定義瞭一個int類型的變量,名為x,緊接著我們將數組d拷貝到s中,同時將d數組中索引為1的元素賦值為10;同理,索引為2的元素賦值為20,因此打印的結果為[{0} {10} {20} {0} {0}],緊接著關於地址值的打印就不做贅述啦

到此這篇關於一文帶你深入瞭解Go語言中切片的奧秘的文章就介紹到這瞭,更多相關Go語言切片內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!

推薦閱讀: