Golang切片的三種簡單使用方式及區別

OldBoy~發表於2018-09-12

概念

切片(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是切片容量(容量必須>=長度)

  1. 通過make方式建立切片可以指定切片大小和容量
  2. 如果沒有給切片的各個元素賦值,那麼就會使用預設值(int、float=>0, strint=>"", bool=>false)
  3. 榮國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]]

 

相關文章