欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺談Golang 切片(slice)擴容機制的原理

 更新時間:2021年06月09日 09:07:12   作者:Linux猿  
我們知道 Golang 切片在容量不足的情況下會進行擴容,擴容的原理是怎樣的呢,具有一定的參考價值,感興趣的小伙伴們可以參考一下

我們知道 Golang 切片(slice) 在容量不足的情況下會進行擴容,擴容的原理是怎樣的呢?是不是每次擴一倍?下面我們結(jié)合源碼來告訴你答案。

一、源碼

Version : go1.15.6  src/runtime/slice.go

//go1.15.6 源碼 src/runtime/slice.go
func growslice(et *_type, old slice, cap int) slice {
 //省略部分判斷代碼
    //計算擴容部分
    //其中,cap : 所需容量,newcap : 最終申請容量
 newcap := old.cap
 doublecap := newcap + newcap
 if cap > doublecap {
  newcap = cap
 } else {
  if old.len < 1024 {
   newcap = doublecap
  } else {
   // Check 0 < newcap to detect overflow
   // and prevent an infinite loop.
   for 0 < newcap && newcap < cap {
    newcap += newcap / 4
   }
   // Set newcap to the requested cap when
   // the newcap calculation overflowed.
   if newcap <= 0 {
    newcap = cap
   }
  }
 } 
 //省略部分判斷代碼
}

二、原理

1. 如果當前所需容量 (cap) 大于原先容量的兩倍 (doublecap),則最終申請容量(newcap)為當前所需容量(cap);

2. 如果<條件1>不滿足,表示當前所需容量(cap)不大于原容量的兩倍(doublecap),則進行如下判斷;

3. 如果原切片長度(old.len)小于1024,則最終申請容量(newcap)等于原容量的兩倍(doublecap);

4. 否則,最終申請容量(newcap,初始值等于 old.cap)每次增加 newcap/4,直到大于所需容量(cap)為止,然后,判斷最終申請容量(newcap)是否溢出,如果溢出,最終申請容量(newcap)等于所需容量(cap);

這樣說大家可能不太明白,來幾個例子:

2.1 實例1

驗證條件1:

package main
 
import "fmt"
 
func main() {
 //第1條中的例子:
 var slice = []int{1, 2, 3}
 var slice1 = []int{4, 5, 6, 7, 8, 9, 10, 11, 12}
 fmt.Printf("slice %v len = %v cap = %v\n", slice, len(slice), cap(slice))
 fmt.Printf("slice1 %v len = %v cap = %v\n", slice1, len(slice1), cap(slice1))
 slice = append(slice, slice1...)
 fmt.Printf("slice %v len = %v cap = %v\n", slice, len(slice), cap(slice))
}

輸出:

[root@localhost test]# go run main.go
slice [1 2 3] len = 3 cap = 3
slice1 [4 5 6 7 8 9 10 11 12] len = 9 cap = 9
slice [1 2 3 4 5 6 7 8 9 10 11 12] len = 12 cap = 12
[root@localhost test]#

在實例1中,所需容量 cap = 9+3 = 12,原容量的兩倍 doublecap = 2 * 3 = 6,滿足 <條件1> 即:所需容量大于原容量的兩倍,所以最終申請容量 newcap = cap = 12。

2.2 實例2

驗證條件2,3:

package main
import "fmt"
 
func main() {
 //第2、3條中的例子:
 var slice = []int{1, 2, 3, 4, 5, 6, 7}
 var slice1 = []int{8, 9}
 fmt.Printf("slice %v len = %v cap = %v\n", slice, len(slice), cap(slice))
 fmt.Printf("slice1 %v len = %v cap = %v\n", slice1, len(slice1), cap(slice1))
 slice = append(slice, slice1...)
 fmt.Printf("slice %v len = %v cap = %v\n", slice, len(slice), cap(slice))
}

 輸出:

[root@localhost test]# go run main.go
slice [1 2 3 4 5 6 7] len = 7 cap = 7
slice1 [8 9] len = 2 cap = 2
slice [1 2 3 4 5 6 7 8 9] len = 9 cap = 14
[root@localhost test]#

在實例2中,所需容量 cap = 7+2 = 9,原容量的兩倍 doublecap = 2*7 = 14,原切片長度 old.len = 7,滿足 <條件2,3>,即: 所需容量小于原容量的兩倍,并且原切片長度 old.len 小于1024,所以,最終申請容量 newcap = doublecap = 14。

2.3 實例3

驗證條件4:

package main
import "fmt"
 
func main() {
 //第2條中的例子:
 var slice []int
 for i := 0; i < 1024; i++ {
  slice = append(slice, i)
 }
 var slice1 = []int{1024, 1025}
 fmt.Printf("slice %v len = %v cap = %v\n", slice, len(slice), cap(slice))
 fmt.Printf("slice1 %v len = %v cap = %v\n", slice1, len(slice1), cap(slice1))
 slice = append(slice, slice1...)
 fmt.Printf("slice %v len = %v cap = %v\n", slice, len(slice), cap(slice))
}

輸出:

[root@localhost test]# go run main.go
slice [0 1 2 3 4 5 6……1017 1018 1019 1020 1021 1022 1023] len = 1024 cap = 1024
slice1 [1024 1025] len = 2 cap = 2
slice [0 1 2 3 4 5 6……1017 1018 1019 1020 1021 1022 1023 1024 1025] len = 1026 cap = 1280
[root@localhost test]#

在實例3中,所需容量 cap = 1024+2 = 1026,doublecap = 2048,  old.len = 1024,滿足 <條件4> ,所以,newcap = 1024 + 1024/4 = 1280。

到此這篇關于淺談Golang 切片(slice)擴容機制的原理的文章就介紹到這了,更多相關Golang 切片擴容機制內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關文章希望大家以后多多支持腳本之家!

相關文章

  • Go語言字符串及strings和strconv包使用實例

    Go語言字符串及strings和strconv包使用實例

    字符串是工作中最常用的,值得我們專門的練習一下,下面這篇文章主要給大家介紹了關于Go語言字符串及strings和strconv包使用的相關資料,文中通過代碼介紹的非常詳細,需要的朋友可以參考下
    2024-06-06
  • Go標準庫http?server的優(yōu)雅關閉深入理解

    Go標準庫http?server的優(yōu)雅關閉深入理解

    這篇文章主要為大家介紹了Go標準庫http?server的優(yōu)雅有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪關閉深入理解
    2024-01-01
  • 解決GO編譯時避免引入外部動態(tài)庫的問題

    解決GO編譯時避免引入外部動態(tài)庫的問題

    最近碰到一個問題,有一個流量采集的組件中使用到了github.com/google/gopacket 這個庫,這個庫使用一切正常,但是唯獨有一個缺點,編譯后的二進制文件依賴于libpcap.so的動態(tài)庫,這篇文章主要介紹了GO編譯時避免引入外部動態(tài)庫的解決方法,需要的朋友可以參考下
    2022-10-10
  • Go語言如何生成PDF文件實例探究

    Go語言如何生成PDF文件實例探究

    這篇文章主要為大家介紹了Go語言生成PDF文件的實例探究,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2024-01-01
  • Golang中處理import自定義包出錯問題的解決辦法

    Golang中處理import自定義包出錯問題的解決辦法

    最近開始使用Go/GoLand在import自定義包時出現(xiàn)各種狀況,下面這篇文章主要給大家介紹了關于Golang中處理import自定義包出錯問題的解決辦法,文中通過圖文介紹的非常詳細,需要的朋友可以參考下
    2023-11-11
  • Golang實現(xiàn)常見排序算法的示例代碼

    Golang實現(xiàn)常見排序算法的示例代碼

    現(xiàn)在的面試真的是越來越卷了,算法已經(jīng)成為了面試過程中必不可少的一個環(huán)節(jié),你如果想進稍微好一點的公司,算法是必不可少的一個環(huán)節(jié)。本文為大家準備了Golang實現(xiàn)常見排序算法的示例代碼,需要的可以參考一下
    2022-05-05
  • Gin+Gorm實現(xiàn)增刪改查的示例代碼

    Gin+Gorm實現(xiàn)增刪改查的示例代碼

    本文介紹了如何使用Gin和Gorm框架實現(xiàn)一個簡單的增刪改查(CRUD)示例,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2024-12-12
  • Golang Map value不可尋址使用指針類型代替示例詳解

    Golang Map value不可尋址使用指針類型代替示例詳解

    這篇文章主要為大家介紹了Golang Map value不可尋址使用指針類型代替示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-11-11
  • GoLand 中設置默認項目文件夾的實現(xiàn)

    GoLand 中設置默認項目文件夾的實現(xiàn)

    本文主要介紹了GoLand 中設置默認項目文件夾的實現(xiàn),默認項目文件夾會在你打開或新建項目時自動預選,避免每次都需要手動導航到目標目錄,感興趣的可以了解一下
    2025-03-03
  • 一文帶你全面掌握Go語言中的正則表達式

    一文帶你全面掌握Go語言中的正則表達式

    正則表達式是一種強大的模式匹配工具,能夠在文本中進行靈活的搜索和替換操作,本文將介紹?Golang?中的正則表達式語法,包括常用的匹配符號、模式修飾符以及示例應用,希望對大家有所幫助
    2023-05-05

最新評論