golang 切片的三種使用方式及區別的說明

概念

切片(slice)是建立在數組之上的更方便,更靈活,更強大的數據結構。切片並不存儲任何元素而隻是對現有數組的引用。

三種方式及細節案例

①定義一個切片,然後讓切片去引用一個已經創建好的數組

package main
import (
    "fmt"
) 
func main() {
    var arr [5]int = [...]int {1, 2, 3, 4, 5}
    var slice = arr[1:3]
    fmt.Println("arr=", arr)
    fmt.Println("slice=", slice)
    fmt.Println("slice len", len(slice))
    fmt.Println("slice cap", cap(slice))
}

②通過make來創建切片

基本語法:var 切片名 []type = make([], len, [cap]);參數說明:type是數據類型、len是大小、cap是切片容量(容量必須>=長度)

通過make方式創建切片可以指定切片大小和容量

如果沒有給切片的各個元素賦值,那麼就會使用默認值(int、float=>0, strint=>””, bool=>false)

榮國make方式創建的切片對應的數組是由make底層維護,對外不可見,也就是隻能通過slice訪問各個元素

package main
import (
    "fmt"
) 
func main() {
    var slice []float64 = make([]float64, 5, 10)
    //沒有給值,默認都是0
    fmt.Println(slice)  //[0 0 0 0 0]
 
    //賦值
    slice[1] = 5
    slice[3] = 10  
    fmt.Println(slice)  //[0 5 0 10 0] 
 
    fmt.Println("slice大小:", len(slice)) //slice大小: 5
    fmt.Println("slice容量:", cap(slice)) //slice容量: 10
}

③定義一個切片,直接就指定具體數組,使用原理類似於make的方式

package main
import (
    "fmt"
) 
func main() {
    var slice []string = []string{"zhangsan", "lisi", "wangwu"}
    fmt.Println("slice=", slice) //slice= [zhangsan lisi wangwu]
    fmt.Println("slice len", len(slice)) //slice len 3
    fmt.Println("slice cap", cap(slice)) //slice cap 3
}

第一種和第二種的區別

第一種方式是直接引用數組,這個數組是事先存在的,程序員可見

第二種方式是通過make來創建切片,make也會創建一個數組,是由切片在底層維護,程序員不可見

補充:零碎案例

package main
import "fmt"
func main() { 
    // 和數組不同的是,切片的長度是可變的。
    // 我們可以使用內置函數make來創建一個長度不為零的切片
    // 這裡我們創建瞭一個長度為3,存儲字符串的切片,切片元素
    // 默認為零值,對於字符串就是""。
    s := make([]string, 3)
    fmt.Println("emp:", s)
 
    // 可以使用和數組一樣的方法來設置元素值或獲取元素值
    s[0] = "a"
    s[1] = "b"
    s[2] = "c"
    fmt.Println("set:", s)
    fmt.Println("get:", s[2])
 
    // 可以用內置函數len獲取切片的長度
    fmt.Println("len:", len(s))
 
    // 切片還擁有一些數組所沒有的功能。
    // 例如我們可以使用內置函數append給切片追加值,然後
    // 返回一個擁有新切片元素的切片。
    // 註意append函數不會改變原切片,而是生成瞭一個新切片,
    // 我們需要用原來的切片來接收這個新切片
    s = append(s, "d")
    s = append(s, "e", "f")
    fmt.Println("apd:", s)
 
    // 另外我們還可以從一個切片拷貝元素到另一個切片
    // 下面的例子就是創建瞭一個和切片s長度相同的新切片
    // 然後使用內置的copy函數來拷貝s的元素到c中。
    c := make([]string, len(s))
    copy(c, s)
    fmt.Println("cpy:", c)
 
    // 切片還支持一個取切片的操作 "slice[low:high]"
    // 獲取的新切片包含元素"slice[low]",但是不包含"slice[high]"
    // 下面的例子就是取一個新切片,元素包括"s[2]","s[3]","s[4]"。
    l := s[2:5]
    fmt.Println("sl1:", l)
 
    // 如果省略low,默認從0開始,不包括"slice[high]"元素
    l = s[:5]
    fmt.Println("sl2:", l)
 
    // 如果省略high,默認為len(slice),包括"slice[low]"元素
    l = s[2:]
    fmt.Println("sl3:", l)
 
    // 我們可以同時聲明和初始化一個切片
    t := []string{"g", "h", "i"}
    fmt.Println("dcl:", t)
 
    // 我們也可以創建多維切片,和數組不同的是,切片元素的長度也是可變的。
    twoD := make([][]int, 3)
    for i := 0; i < 3; i++ {
        innerLen := i + 1
        twoD[i] = make([]int, innerLen)
        for j := 0; j < innerLen; j++ {
            twoD[i][j] = i + j
        }
    }
    fmt.Println("2d: ", twoD)
}

結果

emp: [  ]
set: [a b c]
get: c
len: 3
apd: [a b c d e f]
cpy: [a b c d e f]
sl1: [c d e]
sl2: [a b c d e]
sl3: [c d e f]
dcl: [g h i]
2d:  [[0] [1 2] [2 3 4]]

以上為個人經驗,希望能給大傢一個參考,也希望大傢多多支持WalkonNet。如有錯誤或未考慮完全的地方,望不吝賜教。

推薦閱讀: