用過go語言的親們都知道,slice(中文翻譯爲切片)在編程中常常用到,它表明變長的序列,序列中每一個元素都有相同的類型,相似一個動態數組,利用append能夠實現動態增加,利用slice的特性能夠很容易的切割slice,它們是怎麼實現這些特性的呢?如今咱們來探究一下這些特性的本質是什麼。git
s := []int{1,2,3,4,5} fmt.Println(s) // [1 2 3 4 5]
一個slice類型通常寫做[]T,其中T表明slice中元素的類型;slice的語法和數組很像,只是沒有固定長度而已。github
s := []int{1,2,3,4,5} s = append(s, 6) fmt.Println(s) // [1 2 3 4 5 6]
內置append函數在現有數組的長度 < 1024 時 cap 增加是翻倍的,再往上的增加率則是 1.25,至於爲什麼後面會說。golang
s := []int{1,2,3,4,5,6} s1 := s[0:2] fmt.Println(s1) // [1 2] s2 := s[4:] fmt.Println(s2) // [5 6] s3 := s[:4] fmt.Println(s3) // [1 2 3 4]
package main import "fmt" func main() { slice_1 := []int{1, 2, 3, 4, 5} fmt.Printf("main-->data:\t%#v\n", slice_1) fmt.Printf("main-->len:\t%#v\n", len(slice_1)) fmt.Printf("main-->cap:\t%#v\n", cap(slice_1)) test1(slice_1) fmt.Printf("main-->data:\t%#v\n", slice_1) test2(&slice_1) fmt.Printf("main-->data:\t%#v\n", slice_1) } func test1(slice_2 []int) { slice_2[1] = 6666 // 函數外的slice確實有被修改 slice_2 = append(slice_2, 8888) // 函數外的不變 fmt.Printf("test1-->data:\t%#v\n", slice_2) fmt.Printf("test1-->len:\t%#v\n", len(slice_2)) fmt.Printf("test1-->cap:\t%#v\n", cap(slice_2)) } func test2(slice_2 *[]int) { // 這樣才能修改函數外的slice *slice_2 = append(*slice_2, 6666) }
結果:算法
main-->data: []int{1, 2, 3, 4, 5} main-->len: 5 main-->cap: 5 test1-->data: []int{1, 6666, 3, 4, 5, 8888} test1-->len: 6 test1-->cap: 12 main-->data: []int{1, 6666, 3, 4, 5} main-->data: []int{1, 6666, 3, 4, 5, 6666}
這裏要注意註釋的地方,爲什麼slice做爲值傳遞參數,函數外的slice也被更改了?爲什麼在函數內append不能改變函數外的slice?要回da這些問題就得了解slice內部結構,詳細請看下面.編程
其實slice在Go的運行時庫中就是一個C語言動態數組的實現,在$GOROOT/src/pkg/runtime/runtime.h中能夠看到它的定義:數組
struct Slice { // must not move anything byte* array; // actual data uintgo len; // number of elements uintgo cap; // allocated number of elements };
這個結構有3個字段,第一個字段表示array的指針,就是真實數據的指針(這個必定要注意),因此才常常說slice是數組的引用,第二個是表示slice的長度,第三個是表示slice的容量,注意:len和cap都不是指針。緩存
如今就能夠解釋前面的例子slice做爲函數參數提出的問題: 函數外的slice叫slice_1,函數的參數叫slice_2,當函數傳遞slice_1的時候,其實傳入的確實是slice_1參數的複製,因此slice_2複製了slise_1,但要注意的是slice_2裏存儲的數組的指針,因此當在函數內更改數組內容時,函數外的slice_1的內容也改變了。在函數內用append時,append會自動以倍增的方式擴展slice_2的容量,可是擴展也僅僅是函數內slice_2的長度和容量,slice_1的長度和容量是沒變的,因此在函數外打印時看起來就是沒變。app
在對slice進行append等操做時,可能會形成slice的自動擴容。其擴容時的大小增加規則是:函數
至於爲什麼會這樣,你要看一下golang的源碼就知道了: https://github.com/golang/go/blob/master/src/runtime/slice.go優化
newcap := old.cap if newcap+newcap < cap { newcap = cap } else { for { if old.len < 1024 { newcap += newcap } else { newcap += newcap / 4 } if newcap >= cap { break } } }
在go語言中slice是很靈活的,大部分狀況都能表現的很好,但也有特殊狀況。 當程序要求slice的容量超大而且須要頻繁的更改slice的內容時,就不該該用slice,改用list更合適。