這是一個建立於 的文章,其中的資訊可能已經有所發展或是發生改變。
切片是Go語言的關鍵類型之一,它提供了比數組更多的功能。
樣本1:
package mainimport "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: clen: 3apd: [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]]
數組和切片的定義方式的區別在於[]之中是否有固定長度或者推斷長度標誌符...。
樣本2:
package mainimport "fmt"func main() { s1 := make([]int, 0) test(s1) fmt.Println(s1)}//傳值func test(s []int) { s = append(s, 3) //因為原來分配的空間不夠,所以在另外一個地址又重新分配了空間,所以原始地址的資料沒有變}
輸出結果:
[] //空數組
若改為:
package mainimport "fmt"func main() { s1 := make([]int, 0) s1 = test(s1) fmt.Println(s1)}func test(s []int) []int { s = append(s, 3) return s}
輸出結果:
[3]//正確結果
樣本3:
cap是slice的最大容量,append函數添加元素,如果超過原始slice的容量,會重新分配底層數組。
package mainimport "fmt"func main() { s1 := make([]int, 3, 6) fmt.Println("s1= ", s1, len(s1), cap(s1)) s2 := append(s1, 1, 2, 3) fmt.Println("s1= ", s1, len(s1), cap(s1)) fmt.Println("s2= ", s2, len(s2), cap(s2)) s3 := append(s2, 4, 5, 6) fmt.Println("s1= ", s1, len(s1), cap(s1)) fmt.Println("s2= ", s2, len(s2), cap(s2)) fmt.Println("s3= ", s3, len(s3), cap(s3))}
輸出結果:
s1= [0 0 0] 3 6s1= [0 0 0] 3 6s2= [0 0 0 1 2 3] 6 6s1= [0 0 0] 3 6s2= [0 0 0 1 2 3] 6 6s3= [0 0 0 1 2 3 4 5 6] 9 12
樣本4:
指向同一底層數組的slice之間copy時,允許存在重疊。copy數組時,受限於src和dst數組的長度最小值。
package mainimport "fmt"func main() { s1 := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9} s2 := make([]int, 3, 20) var n int n = copy(s2, s1) fmt.Println(n, s2, len(s2), cap(s2)) s3 := s1[4:6] fmt.Println(n, s3, len(s3), cap(s3)) n = copy(s3, s1[1:5]) fmt.Println(n, s3, len(s3), cap(s3))}
輸出結果:
3 [0 1 2] 3 203 [4 5] 2 62 [1 2] 2 6