主頁(yè) > 知識(shí)庫(kù) > 淺談golang slice 切片原理

淺談golang slice 切片原理

熱門標(biāo)簽:地方門戶網(wǎng)站 呼叫中心市場(chǎng)需求 服務(wù)外包 百度競(jìng)價(jià)排名 網(wǎng)站排名優(yōu)化 Linux服務(wù)器 AI電銷 鐵路電話系統(tǒng)

slice介紹

數(shù)組的長(zhǎng)度在定義之后無(wú)法再次修改;數(shù)組是值類型,每次傳遞都將產(chǎn)生一份副本。顯然這種數(shù)據(jù)結(jié)構(gòu)無(wú)法完全滿足開發(fā)者的真實(shí)需求。在初始定義數(shù)組時(shí),我們并不知道需要多大的數(shù)組,因此我們就需要“動(dòng)態(tài)數(shù)組”。在Go里面這種數(shù)據(jù)結(jié)構(gòu)叫slice,slice并不是真正意義上的動(dòng)態(tài)數(shù)組,而是一個(gè)引用類型。slice總是指向一個(gè)底層array,slice的聲明也可以像array一樣,只是不需要長(zhǎng)度,它是可變長(zhǎng)的,可以隨時(shí)往slice里面加數(shù)據(jù)。

初看起來(lái),數(shù)組切片就像一個(gè)指向數(shù)組的指針,實(shí)際上它擁有自己的數(shù)據(jù)結(jié)構(gòu),而不僅僅是個(gè)指針。數(shù)組切片的數(shù)據(jù)結(jié)構(gòu)可以抽象為以下3個(gè)變量:

1.一個(gè)指向原生數(shù)組的指針(point):指向數(shù)組中slice指定的開始位置;
2.?dāng)?shù)組切片中的元素個(gè)數(shù)(len):即slice的長(zhǎng)度;
3.?dāng)?shù)組切片已分配的存儲(chǔ)空間(cap):也就是slice開始位置到數(shù)組的最后位置的長(zhǎng)度。

從底層實(shí)現(xiàn)的角度來(lái)看,數(shù)組切片實(shí)際上仍然使用數(shù)組來(lái)管理元素,基于數(shù)組,數(shù)組切片添加了一系列管理功能,可以隨時(shí)動(dòng)態(tài)擴(kuò)充存放空間,并且可以被隨意傳遞而不會(huì)導(dǎo)致所管理的元素被重復(fù)復(fù)制。

slice聲明

聲明slice時(shí)方括號(hào)[]內(nèi)沒有任何數(shù)據(jù)
聲明一個(gè)元素類型為int的slice
var mySlice []int    聲明兩個(gè)元素類型為byte的slice   

golang 中的 slice 非常強(qiáng)大,讓數(shù)組操作非常方便高效。在開發(fā)中不定長(zhǎng)度表示的數(shù)組全部都是 slice 。但是很多同學(xué)對(duì) slice 的模糊認(rèn)識(shí),造成認(rèn)為golang中的數(shù)組是引用類型,結(jié)果就是在實(shí)際開發(fā)中碰到很多坑,以至于出現(xiàn)一些莫名奇妙的問(wèn)題,數(shù)組中的數(shù)據(jù)丟失了。

下面我們就開始詳細(xì)理解下 slice ,理解后會(huì)對(duì)開發(fā)出高效的程序非常有幫助。

這個(gè)是 slice 的數(shù)據(jù)結(jié)構(gòu),它很簡(jiǎn)單,一個(gè)指向真實(shí) array 地址的指針 ptr ,slice 的長(zhǎng)度 len 和容量 cap 。

其中 len 和 cap 就是我們?cè)谡{(diào)用 len(slice) 和 cap(slice) 返回的值。

我們來(lái)按照 slice 的數(shù)據(jù)結(jié)構(gòu)定義來(lái)解析出 ptr, len, cap

// 按照上圖定義的數(shù)據(jù)結(jié)構(gòu)
type Slice struct {
  ptr  unsafe.Pointer    // Array pointer
  len  int        // slice length
  cap   int        // slice capacity
}

下面寫一個(gè)完整的程序,嘗試把golang中slice的內(nèi)存區(qū)域轉(zhuǎn)換成我們定義的 Slice 進(jìn)行解析

package main

import (
  "fmt"
  "unsafe"
)

// 按照上圖定義的數(shù)據(jù)結(jié)構(gòu)
type Slice struct {
  ptr unsafe.Pointer // Array pointer
  len int      // slice length
  cap int      // slice capacity
}

// 因?yàn)樾枰羔樣?jì)算,所以需要獲取int的長(zhǎng)度
// 32位 int length = 4
// 64位 int length = 8
var intLen = int(unsafe.Sizeof(int(0)))

func main() {
  s := make([]int, 10, 20)

  // 利用指針讀取 slice memory 的數(shù)據(jù)
  if intLen == 4 { // 32位
    m := *(*[4 + 4*2]byte)(unsafe.Pointer(s))
    fmt.Println("slice memory:", m)
  } else { // 64 位
    m := *(*[8 + 8*2]byte)(unsafe.Pointer(s))
    fmt.Println("slice memory:", m)
  }

  // 把slice轉(zhuǎn)換成自定義的 Slice struct
  slice := (*Slice)(unsafe.Pointer(s))
  fmt.Println("slice struct:", slice)
  fmt.Printf("ptr:%v len:%v cap:%v \n", slice.ptr, slice.len, slice.cap)
  fmt.Printf("golang slice len:%v cap:%v \n", len(s), cap(s))

  s[0] = 0
  s[1] = 1
  s[2] = 2

  // 轉(zhuǎn)成數(shù)組輸出
  arr := *(*[3]int)(unsafe.Pointer(slice.ptr))
  fmt.Println("array values:", arr)

  // 修改 slice 的 len
  slice.len = 15
  fmt.Println("Slice len: ", slice.len)
  fmt.Println("golang slice len: ", len(s))
}

運(yùn)行一下查看結(jié)果

$ go run slice.go

slice memory: [0 64 6 32 200 0 0 0 10 0 0 0 0 0 0 0 20 0 0 0 0 0 0 0]
slice struct: {0xc820064000 10 20}
ptr:0xc820064000 len:10 cap:20
golang slice len:10 cap:20
array values: [0 1 2]
Slice len: 15
golang slice len: 15

看到了,golang slice 的memory內(nèi)容,和自定義的 Slice 的值,還有按照 slice 中的指針指向的內(nèi)存,就是實(shí)際 Array 數(shù)據(jù)。當(dāng)修改了 slice 中的len, len(s) 也變了。

接下來(lái)結(jié)合幾個(gè)例子,了解下slice一些用法

聲明一個(gè)Array通常使用 make ,可以傳入2個(gè)參數(shù),也可傳入3個(gè)參數(shù),第一個(gè)是數(shù)據(jù)類型,第二個(gè)是 len ,第三個(gè)是 cap 。如果不穿入第三個(gè)參數(shù),則 cap=len ,append 可以用來(lái)向數(shù)組末尾追加數(shù)據(jù)。

這是一個(gè) append 的測(cè)試

// 每次cap改變,指向array的ptr就會(huì)變化一次
s := make([]int, 1)

fmt.Printf("len:%d cap: %d array ptr: %v \n", len(s), cap(s), *(*unsafe.Pointer)(unsafe.Pointer(s)))

for i := 0; i  5; i++ {
  s = append(s, i)
  fmt.Printf("len:%d cap: %d array ptr: %v \n", len(s), cap(s), *(*unsafe.Pointer)(unsafe.Pointer(s)))
}

fmt.Println("Array:", s)

運(yùn)行結(jié)果

len:1 cap: 1 array ptr: 0xc8200640f0
len:2 cap: 2 array ptr: 0xc820064110
len:3 cap: 4 array ptr: 0xc8200680c0
len:4 cap: 4 array ptr: 0xc8200680c0
len:5 cap: 8 array ptr: 0xc82006c080
len:6 cap: 8 array ptr: 0xc82006c080
Array: [0 0 1 2 3 4]

看出來(lái)了吧,每次cap改變的時(shí)候指向array內(nèi)存的指針都在變化。當(dāng)在使用 append 的時(shí)候,如果 cap==len 了這個(gè)時(shí)候就會(huì)新開辟一塊更大內(nèi)存,然后把之前的數(shù)據(jù)復(fù)制過(guò)去。

實(shí)際go在append的時(shí)候放大cap是有規(guī)律的。在 cap 小于1024的情況下是每次擴(kuò)大到 2 * cap ,當(dāng)大于1024之后就每次擴(kuò)大到 1.25 * cap 。所以上面的測(cè)試中cap變化是 1, 2, 4, 8

在實(shí)際使用中,我們最好事先預(yù)期好一個(gè)cap,這樣在使用append的時(shí)候可以避免反復(fù)重新分配內(nèi)存復(fù)制之前的數(shù)據(jù),減少不必要的性能消耗。

創(chuàng)建切片

s := []int{1, 2, 3, 4, 5}
fmt.Printf("len:%d cap: %d array ptr: %v \n", len(s), cap(s), *(*unsafe.Pointer)(unsafe.Pointer(s)))
fmt.Println("Array:", s)

s1 := s[1:3]
fmt.Printf("len:%d cap: %d array ptr: %v \n", len(s1), cap(s1), *(*unsafe.Pointer)(unsafe.Pointer(s1)))
fmt.Println("Array", s1)

運(yùn)行結(jié)果

len:5 cap: 5 array ptr: 0xc820012210
Array: [1 2 3 4 5]
len:2 cap: 4 array ptr: 0xc820012218
Array [2 3]

在一個(gè)切片基礎(chǔ)上創(chuàng)建新的切片 s1 ,新切片的 ptr 指向的就是 s1[0] 數(shù)據(jù)的內(nèi)存地址??梢钥吹街羔樀刂?0xc820012210 與 0xc820012218 相差 8byte 正好是一個(gè)int類型長(zhǎng)度,cap也相應(yīng)的變?yōu)?

就寫到這里了,總結(jié)一下,切片的結(jié)構(gòu)是指向數(shù)據(jù)的指針,長(zhǎng)度和容量。復(fù)制切片,或者在切片上創(chuàng)建新切片,切片中的指針都指向相同的數(shù)據(jù)內(nèi)存區(qū)域。

知道了切片原理就可以在開發(fā)中避免出現(xiàn)錯(cuò)誤了,希望這篇博客可以給大家?guī)?lái)幫助。也希望大家多多支持腳本之家。

您可能感興趣的文章:
  • golang常用手冊(cè)之切片(Slice)原理
  • Golang slice切片操作之切片的追加、刪除、插入等
  • 理解Golang中的數(shù)組(array)、切片(slice)和map
  • golang-切片slice的創(chuàng)建方式

標(biāo)簽:蘭州 湘潭 湖南 衡水 仙桃 崇左 銅川 黃山

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《淺談golang slice 切片原理》,本文關(guān)鍵詞  ;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問(wèn)題,煩請(qǐng)?zhí)峁┫嚓P(guān)信息告之我們,我們將及時(shí)溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無(wú)關(guān)。
  • 相關(guān)文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話咨詢

    • 400-1100-266