主頁 > 知識庫 > Golang中的Slice與數(shù)組及區(qū)別詳解

Golang中的Slice與數(shù)組及區(qū)別詳解

熱門標(biāo)簽:澳門防封電銷卡 賺地圖標(biāo)注的錢犯法嗎 福州鐵通自動外呼系統(tǒng) 長沙ai機(jī)器人電銷 烏魯木齊人工電銷機(jī)器人系統(tǒng) 廣東語音外呼系統(tǒng)供應(yīng)商 濮陽自動外呼系統(tǒng)代理 智能電銷機(jī)器人營銷 地圖標(biāo)注測試

在golang中有數(shù)組和Slice兩種數(shù)據(jù)結(jié)構(gòu),Slice是基于數(shù)組的實現(xiàn),是長度動態(tài)不固定的數(shù)據(jù)結(jié)構(gòu),本質(zhì)上是一個對數(shù)組字序列的引用,提供了對數(shù)組的輕量級訪問。那么我們今天就給大家詳細(xì)介紹下Golang中的Slice與數(shù)組,

1.Golang中的數(shù)組

    數(shù)組是一種具有固定長度的基本數(shù)據(jù)結(jié)構(gòu),在golang中與C語言一樣數(shù)組一旦創(chuàng)建了它的長度就不允許改變,數(shù)組的空余位置用0填補(bǔ),不允許數(shù)組越界。

     數(shù)組的一些基本操作:

     1.創(chuàng)建數(shù)組:

func main() {
 var arr1 = [...]int{1,2,3,4} //[...]默認(rèn)為元素的數(shù)量即為數(shù)組的長度
 fmt.Println(len(arr1)) //4
 arr1[4] = 5 //panic 數(shù)組越界
 fmt.Println(arr1)
 var arr2 = [10]int{1,2,3,4}
 fmt.Println(arr2) //[1 2 3 4 0 0 0 0 0 0]
}

  2.數(shù)組是值拷貝傳遞:

func main() {
 var arr = [10]int{4,5,7,11,8,9}
 fmt.Println(arr) //[4,5,7,11,8,9,0,0,0,0]
 //驗證數(shù)組是值拷貝傳遞
 AddOne(arr)
 fmt.Println(arr) //[4,5,7,11,8,9,0,0,0,0]
}
func AddOne(arr [10]int){
 arr[9] = 999999
 fmt.Println(arr) //[4,5,7,11,8,9,0,0,0,999999]
}

2.Golang中的切片(slice) 

     1.首先看看slice的源碼結(jié)構(gòu):

type slice struct {
 array unsafe.Pointer
 len int
 cap int
}

  slice是一個特殊的引用類型,但是它自身也是個結(jié)構(gòu)體

       屬性len表示可用元素數(shù)量,讀寫操作不能超過這個限制,不然就會panic

       屬性cap表示最大擴(kuò)張容量,當(dāng)然這個擴(kuò)張容量也不是無限的擴(kuò)張,它是受到了底層數(shù)組array的長度限制,超出了底層array的長度就會panic

     2.slice的創(chuàng)建:

func main() {
  var arr = [...]int{0,1,2,3,4,5,6}
  slice1 := arr[1:4:5] //{low:high:max} 最多再擴(kuò)張一個元素
  //max超出 len(arr)
  //slice2 := arr[1:4:7] //panic
  fmt.Println(slice1) //[1,2,3]
  slice3 := slice1[1:3:4] //[2,3] 大于4會panic
  fmt.Println(slice3)
}

 上面代碼中創(chuàng)建了一個長度為7的數(shù)組arr,同時創(chuàng)建一個基于數(shù)組arr的切片slice1,切片引用了數(shù)組的index=1到index=3之間的元素,同時也允許切片最大擴(kuò)張1個元素大小的空間。如果這個擴(kuò)張空間大于7那么程序就會panic。最后創(chuàng)建了一個基于slice1延申的一個切片slice2,它引用了切片的index=1到index=3之間的元素,由于slice1最大擴(kuò)容1個元素,因此slice2也最多擴(kuò)容一個元素,超過了會panic。

    創(chuàng)建基于底層數(shù)組的slice,其cap取值在: len=cap=len(arr)之間

    創(chuàng)建基于一個切片的slice,其cap取值在: len(slice1)=cap=cap(slice1)之間

   3.slice使用make創(chuàng)建

func main() {
  var slice = make([]int,3,5) //len=3,cap=5
  fmt.Println(slice)  //[0,0,0]
  slice2:=slice[:5]  //slice實現(xiàn)了對slice的擴(kuò)容,切片長度變?yōu)?
  fmt.Println(slice2) //[0,0,0,0,0]
}

   4.切片作為參數(shù)傳遞

func main() {
  var slice = make([]int,3,5) //len=3,cap=5
  fmt.Println(slice)  //[0,0,0]
  slice2:=slice[:5]  //slice實現(xiàn)了對slice的擴(kuò)容,切片長度變?yōu)?
  fmt.Println(slice2) //[0,0,0,0,0]
  slice[0] = 999  //這里slice和slice的index=0位置都是999 因為他們引用的底層數(shù)組的index=0位置都是999
  fmt.Println(slice)
  fmt.Println(slice2)
  AddOne(slice) //[8888,0,0]
  fmt.Println(slice) //[8888,0,0]
  fmt.Println(slice2) //[8888,0,0,0]
}
func AddOne(s []int){
 s[0] = 8888
 fmt.Println(s)
}

  因為切片是個引用類型,所以它作為參數(shù)傳遞給函數(shù),函數(shù)操作的實質(zhì)是底層數(shù)組

3.Golang中的切片追加append()

func main() {
  var arr = [...]int{1,2,3,4}
  fmt.Println(arr) //[1,2,3,4]
  slice := arr[:]
  fmt.Println(slice) //[1,2,3,4]
  slice = append(slice,[]int{5,6,7}...) //此時slice的引用地址已經(jīng)發(fā)生改變了,它引用的底層數(shù)組再也不是arr了,而是一個新的數(shù)組newarr[1,2,3,4,5,6,7]
  fmt.Println(slice) //[1,2,3,4,5,6,7]
  //驗證slice引用的地址已經(jīng)發(fā)生改變
  slice[0] = 666
  fmt.Println(arr) //[1,2,3,4]
  fmt.Println(slice) //[666,2,3,4,5,6,7]
}

  這里由于slice進(jìn)行追加的元素超出了原來數(shù)組的大小,因此go內(nèi)部會幫我們創(chuàng)建一個新的底層數(shù)組,而slice的引用地址不再是arr了,變成了新創(chuàng)建的數(shù)組。

       還有一種情況就是當(dāng)slice進(jìn)行追加的時候沒有超出原來數(shù)組的大小的時候,其引用地址沒有發(fā)生改變。

func main() {
  var arr = [6]int{1,2,3,4}
  fmt.Println(arr) //[1,2,3,4,0,0]
  slice := arr[:4]
  fmt.Println(slice) //[1,2,3,4]
  slice = append(slice,5)
  fmt.Println(arr) //[1,2,3,4,5,0]
  fmt.Println(slice) //[1,2,3,4,5]
}

4.總結(jié)

    (1)go是有數(shù)組的,只是平時用切片比較多。數(shù)組大小一旦創(chuàng)建就不能改變,數(shù)組長度大于元素個數(shù)的時候會用0補(bǔ)位,這跟其他語言是相通的。

     (2)切片slice可以看作是對數(shù)組的一切操作,它是一個引用數(shù)據(jù)類型,其數(shù)據(jù)結(jié)構(gòu)包括底層數(shù)組的地址,以及元素可操作長度len或可擴(kuò)容長度cap。

     (3)要想突破slice的擴(kuò)容cap限制進(jìn)行無限擴(kuò)容就需要使用append()函數(shù)進(jìn)行操作。如果append追加的元素后slice的總長度不超過底層數(shù)組的總長度,那么slice引用的地址不會發(fā)生改變,反之引用地址會           變成新的數(shù)組的地址。

     (4)slice是一個抽象的概念,它存在的意義在于方便對一個順序結(jié)構(gòu)進(jìn)行一些方便操作,例如查找,排序,追加等等,這個類似于python的list。

下面看下golang 數(shù)組和slice 的區(qū)別

golang 數(shù)組和切片的區(qū)別

數(shù)組: 長度不可變,初始化的時候聲明長度

slice 長度可變

var a [32] int

var b [3][5] int

a和b的類型不一樣

slice 創(chuàng)建的時候可以不指定長度。

總結(jié)

到此這篇關(guān)于Golang中的Slice與數(shù)組及區(qū)別詳解的文章就介紹到這了,更多相關(guān)golang slice 數(shù)據(jù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

您可能感興趣的文章:
  • 深入解析Go語言編程中slice切片結(jié)構(gòu)
  • go 判斷兩個 slice/struct/map 是否相等的實例
  • golang語言如何將interface轉(zhuǎn)為int, string,slice,struct等類型
  • Go 中 slice 的 In 功能實現(xiàn)探索
  • Go語言中slice作為參數(shù)傳遞時遇到的一些“坑”
  • 深入理解go slice結(jié)構(gòu)

標(biāo)簽:貴陽 德州 西雙版納 慶陽 調(diào)研邀請 廣西 阿克蘇 太原

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《Golang中的Slice與數(shù)組及區(qū)別詳解》,本文關(guān)鍵詞  Golang,中的,Slice,與,數(shù)組,;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 下面列出與本文章《Golang中的Slice與數(shù)組及區(qū)別詳解》相關(guān)的同類信息!
  • 本頁收集關(guān)于Golang中的Slice與數(shù)組及區(qū)別詳解的相關(guān)信息資訊供網(wǎng)民參考!
  • 推薦文章