Golang原理分析:切片(slice)原理及扩容机制
创始人
2024-03-17 17:39:32
0

在这里插入图片描述

《Go语言精进之路》切片相关章节学习笔记及实验。

1.切片原理

说切片之前,先看看Go语言数组。Go数组是一个固定长度的、容纳同构类型元素的连续序列,因此Go数组类型具有两个属性:长度及类型:

var a [1]int
var b [2]byte
var c [3]string

数组的特性是声明即定长,无法原地进行扩容,所以Golang中一般来说需要使用到数组的场合都会使用切片来替代。

切片的特性:

  • 不定长,能随着存储容量进行自动扩容
  • 提供append函数,能够方便地添加值

切片的本质是引用底层数组头指针+当前切片长度+底层数组大小:即array、len和cap:

  • $goroot/src/runtime/slice.go
type slice struct {array unsafe.Pointer    // 底层数组头指针len   int               // 当前切片长度 cap   int               // 底层数组最大容量
}

使用时通过make关键字来进行初始化切片分配一个新的底层数组,或者是基于指定的数组进行切片化:

  • make关键字,创建一个长度为5的切片,如果没有指定cap参数则默认与长度一致为5(底层数组长度也为5)
s := make([]byte,5)

在这里插入图片描述

  • 直接通过已有数组创建切片,array指向底层数组的指定位置作为头部,从头部开始计算len及cap:
u := [10]byte{11,12,13,14,15,16,17,18,19,20} // 长度为10的数组
s := u[3:7] // 从数组的第3个元素至第6个元素建立切片(前闭后开区间)

在这里插入图片描述

  • 底层数组可以被多个切片共享,如果切片共享了底层数组同一值,那么切片1进行修改后,切片2读取该值时也会读取到修改的结果:
    在这里插入图片描述

2.扩容机制

切片能够在底层数组容量不够时进行扩容,每次append时如果容量不足,会创建2倍大小于老数组的新数组,并复制老数组数据至新的数组中:

func main() {var s []int                // 初始化时,底层数组为数组为nils = append(s, 11)          fmt.Println(len(s), cap(s)) // 1 1s = append(s, 12) fmt.Println(len(s), cap(s)) // 2 2s = append(s, 13)fmt.Println(len(s), cap(s)) // 3 4s = append(s, 14)fmt.Println(len(s), cap(s)) // 4 4s = append(s, 15)fmt.Println(len(s), cap(s)) // 5 8
}

在这里插入图片描述

  • 扩容存在损耗,所以尽可能在已知数据最长度的情况下声明切片的cap,避免频繁库容切换底层数组带来的性能损耗:
func BenchmarkAppendZeroSlice(b *testing.B) {maxNum := 10000for i := 0; i < b.N; i++ {// 空切片var zeroSlice []intfor j := 0; j < maxNum; j++ {zeroSlice = append(zeroSlice, j)}}
}func BenchmarkAppendCapSlice(b *testing.B) {maxNum := 10000for i := 0; i < b.N; i++ {// 声明cap的切片capSlice := make([]int, 0, maxNum)for j := 0; j < maxNum; j++ {capSlice = append(capSlice, j)}}
}

在这里插入图片描述

从基准测试结果可以看出,声明了cap的切片能避免频繁创建底层数组及复制,单轮append时间为18847ns,相较于零值切片的82081ns提升了4.3倍性能。

相关内容

热门资讯

原创 中... 12月25日,家纺企业富安娜披露了关于中信证券固定收益类理财产品逾期兑付的进展公告。公告显示,公司近...
封关临近!海南自贸港政策红利释... 交易所数据显示,2025年12月26日09时47分,京粮控股当前价格为8.92元,涨幅为9.99%,...
字节跳动通报:120名员工被辞... 12月25日,字节跳动披露2025年三季度内部违规案例的处理情况。通报显示,三季度共有120名员工因...
上亿理财难收回,家纺龙头富安娜... 12月25日晚,家纺龙头企业深圳市富安娜家居用品股份有限公司(以下简称富安娜,002327.SZ)发...
华院计算取得法律要素图谱辅助类... 国家知识产权局信息显示,华院计算技术(上海)股份有限公司取得一项名为“一种法律要素图谱辅助类案推荐方...
严重违背人伦底线,犯罪手段特别... 据“遵义审判”消息,2025年12月26日,贵州省遵义市中级人民法院依法对被告人刘仲杰故意杀人案进行...
原创 死... 死刑判决能否抚平受害者家属的创伤?法律与心理的双重拷问 当一纸死刑判决书尘埃落定,法庭外的受害者家属...
贵州工会“一函两书”典型案例⑩... 编者按:“一函两书”制度,是工会组织开展劳动法律监督,联合检察院、法院、人社等部门提醒用人单位落实好...
建元信托4.98亿诉讼纠纷再起... 今年以来,建元信托遭遇11起诉讼案件,包括2起大额诉讼。 文/每日财报 楚风 建元信托的诉讼风险仍...
【学政策·人事人才篇】取得哪些... 来源:人力资源和社会保障部微信公众号