日韩性视频-久久久蜜桃-www中文字幕-在线中文字幕av-亚洲欧美一区二区三区四区-撸久久-香蕉视频一区-久久无码精品丰满人妻-国产高潮av-激情福利社-日韩av网址大全-国产精品久久999-日本五十路在线-性欧美在线-久久99精品波多结衣一区-男女午夜免费视频-黑人极品ⅴideos精品欧美棵-人人妻人人澡人人爽精品欧美一区-日韩一区在线看-欧美a级在线免费观看

歡迎訪問 生活随笔!

生活随笔

當前位置: 首頁 > 编程资源 > 编程问答 >内容正文

编程问答

详解go语言的array和slice 【二】

發布時間:2023/12/9 编程问答 35 豆豆
生活随笔 收集整理的這篇文章主要介紹了 详解go语言的array和slice 【二】 小編覺得挺不錯的,現在分享給大家,幫大家做個參考.

  上一篇 ?詳解go語言的array和slice 【一】已經講解過,array和slice的一些基本用法,使用array和slice時需要注意的地方,特別是slice需要注意的地方比較多。上一篇的最后講解到創建新的slice時使用第三個索引來限制slice的容量,在操作新slice時,如果新slice的容量大于長度時,添加新元素依然后使源的相應元素改變。這一篇里我會講解到如何避免這些問題,以及迭代、和做為方法參數方面的知識點。

slice的長度和容量設置為同一個值

  如果在創建新的slice時我們把他的長度和容量的值設置為樣的值,那么在append新元素時,底層會創建一個新的array并把之前的值復制過去。這樣就不會影響之前共同的底層array了。

// 創建一個容量和長度均為6的sliceslice1 := []int{5, 23, 10, 2, 61, 33}// 對slices1進行切片,長度為2容量為3slice2 := slice1[1:3:3]fmt.Println("cap", cap(slice2))fmt.Println("slice2", slice2)//修改一個共同指向的元素//兩個slice的值都會修改slice2[0] = 11111fmt.Println("slice1", slice1)fmt.Println("slice2", slice2)// 增加一個元素slice2 = append(slice2, 55555)fmt.Println("slice1: ", slice1)fmt.Println("slice2: ", slice2)

  輸出結果

cap 2 slice2 [23 10] slice1 [5 11111 10 2 61 33] slice2 [11111 10] slice1: [5 11111 10 2 61 33] slice2: [11111 10 55555]

  代碼中的長度和容量是一樣的,長度和容量的計算公式看我的上一篇博客。增加一個元素后,原來共同指向的底層數據是沒有變的。因為slice2的底層array被重新賦值了。

迭代slice

  go語言內置一個關鍵字range用于迭代集合,當然他也可以迭代slice,也可以使用? _? 來忽略我們不關心的元素,但是如果只關心index則不需這么寫 for index,_ := range slice1。下在給出完整代碼

// 創建一個容量和長度均為6的sliceslice1 := []int{5, 23, 10, 2, 61, 33}for index, value := range slice1 {fmt.Println("index: ", index, " value: ", value)}// 可以忽略我們不關心的元素// 只關心valuefor _, value := range slice1 {fmt.Println("value ", value)}// 只關心index, 可以不用 _for index := range slice1 {fmt.Println("index: ", index)}

?

?  需要注意的是rang 迭代的value值并是一個復本,我們可以對比一下迭代的value和原slice內相應index下value的地址:

// 創建一個容量和長度均為6的sliceslice1 := []int{5, 23, 10, 2, 61, 33}for index, value := range slice1 {fmt.Println("index: ", index, " value address : ", &value, " slice1 value address", &slice1[index])}

  輸出結果

index: 0 value address : 0xc04204e088 slice1 value address 0xc04206a030 index: 1 value address : 0xc04204e088 slice1 value address 0xc04206a038 index: 2 value address : 0xc04204e088 slice1 value address 0xc04206a040 index: 3 value address : 0xc04204e088 slice1 value address 0xc04206a048 index: 4 value address : 0xc04204e088 slice1 value address 0xc04206a050 index: 5 value address : 0xc04204e088 slice1 value address 0xc04206a058

  slice1中value的地址是不斷變化的。而迭代的value值的地址沒有變化,這是因為value是一個變量,為次迭代的時候賦不同的值。我們把代碼寫成下面這樣,你就清楚了

var index, value intfor index, value = range slice1 {fmt.Println("index: ", index, &index, " value address : ", &value, " slice1 value address", &slice1[index])}

?  除了使用rang 也可以使用傳統的for循環來做迭代

slice1 := []int{5, 23, 10, 2, 61, 33}for i, len := 1, len(slice1); i < len; i++ {fmt.Println("index: ", i, " value:", slice1[i])}

?

?slice作為方法參數

?  由于slice的特殊結構,有一個指針指向一個數組

s := make([]int, 2, 5)fmt.Println("len: ", len(s))fmt.Println("cap: ", cap(s))s = append(s, 2)s[0] = 12

?

  所以,slice做為方法的參數傳遞時,只會復制slice本身而不會復制slice底層的array.如果我們創建一個int類型有100萬長度的slice ,把他傳遞給一個方法時,只需要復制24個字節就夠了。指針需要8個,長度和容量都是8個。

const size int = 1000 * 1000func main() {slice0 := make([]int, size)fmt.Println("slice0 len: ", len(slice0), " cap :", cap(slice0))doSomeThing(slice0) }func doSomeThing(s []int) {fmt.Println(len(s)) }

?

?

轉載于:https://www.cnblogs.com/li-peng/p/7541554.html

總結

以上是生活随笔為你收集整理的详解go语言的array和slice 【二】的全部內容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網站內容還不錯,歡迎將生活随笔推薦給好友。