你还在以为golang切片扩容是1/4的增加吗?其实早就已经变啦!

前言

  1. 最近看go语言的切片原理时发现实际的扩容规则和一些博客和书上讲的不一致,所以看看各版本的源码中切片的扩容到底是怎么做的,看源码是最快捷且准确的方式,
    go语言的源码可以在仓库https://github.com/golang/go上找到,当然也可以在官网下载下来再去翻看,这里不再赘述

切片扩容对比

切片的扩容代码主要在各版本src/runtime/slice.go文件中,主要的扩容代码可以看growslice这个函数

Go1.17(这个版本中的扩容也是目前大部分书籍和博客中讲到的切片扩容内容)

func growslice(et *_type, old slice, cap int) slice {
	...

	newcap := old.cap
	doublecap := newcap + newcap
	if cap > doublecap {
		newcap = cap
	} else {
		if old.cap < 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
			}
		}
	}

	...内存对齐...

	return slice{p, old.len, newcap}
}

1. 这里的主要扩容核心逻辑为:
    1. 如果新申请的容量(cap)大于旧容量(old.cap)的两倍,则最终容量(newcap)是新申请的容量(cap)
    2. 如果旧切片的长度小于1024,则最终容量是旧容量的2倍,即newcap=doublecap
    3. 如果旧切片的长度大于或者等于1024,则最终容量从旧容量开始循环增加原来的1/4,直到最终容量大于或等于新申请的容量
    4. 如果最终容量计算值溢出,即超过了int的最大范围,则最终容量就是新申请的容量

Go1.18(这个是目前实际的扩容规则,虽然在1.20中入参变了,但是核心的扩容逻辑没有变,所以还是以1.18举例)

func growslice(et *_type, old slice, cap int) slice {
	...

	newcap := old.cap
	doublecap := newcap + newcap
	if cap > doublecap {
		newcap = cap
	} else {
		const threshold = 256
		if old.cap < threshold {
			newcap = doublecap
		} else {
			// Check 0 < newcap to detect overflow
			// and prevent an infinite loop.
			for 0 < newcap && newcap < cap {
				// Transition from growing 2x for small slices
				// to growing 1.25x for large slices. This formula
				// gives a smooth-ish transition between the two.
				newcap += (newcap + 3*threshold) / 4
			}
			// Set newcap to the requested cap when
			// the newcap calculation overflowed.
			if newcap <= 0 {
				newcap = cap
			}
		}
	}

	...内存对齐...

	return slice{p, old.len, newcap}
}

1. 这里的扩容核心逻辑是:
    1. 如果新申请的容量(cap)大于旧容量(old.cap)的两倍,则最终容量(newcap)是新申请的容量(cap)
    2. 定义了一个阈值为256,如果旧容量小于256,那么最终容量(newcap)是旧容量(old.cap)的两倍
    3. 如果旧容量(old.cap)大于或等于256,那么最终容量开始从(旧容量+3*256)/4开始循环,直到最终容量大于等于申请的容量
    4. 如果最终容量计算值溢出,即超过了int的最大范围,则最终容量就是新申请的容量

结语

可以看到切片的扩容规则在1.18进行了修改,直到本文发出(截至1.21为止),当前的扩容逻辑和1.18是大致一致的
如果在使用过程中发现与书中或博客中不一致为问题,还是以源码为主,源码是不会骗人的!!!

  • 10
    点赞
  • 6
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Golang中,切片扩容是通过内置函数append来实现的。具体实现方法是使用slice结合golang内置方法append进行动态扩容。\[1\]当切片的容量不足以容纳新的元素时,append函数会创建一个新的底层数组,并将原来的元素复制到新的数组中。然后,将新的元素添加到新的数组中,并返回一个新的切片。这样就实现了切片扩容切片的底层也是在连续的内存块中分配的,所以切片还能获得索引、迭代以及为垃圾回收优化的好处。\[2\]切片是一个非常小的对象,它是对底层的数组进行了抽象,并且提供了相关的操作方法。它拥有三个字段,分别是指向底层数组的指针、长度和容量。通过对切片再次切片,可以缩小一个切片的大小。\[3\]所以,通过使用append函数和切片的特性,可以实现切片的动态扩容。 #### 引用[.reference_title] - *1* *3* [golang slice扩容机制](https://blog.csdn.net/qq_52696089/article/details/126171790)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^koosearch_v1,239^v3^insert_chatgpt"}} ] [.reference_item] - *2* [Golang Slice切片如何扩容](https://blog.csdn.net/moer0/article/details/122933748)[target="_blank" data-report-click={"spm":"1018.2226.3001.9630","extra":{"utm_source":"vip_chatgpt_common_search_pc_result","utm_medium":"distribute.pc_search_result.none-task-cask-2~all~insert_cask~default-1-null.142^v91^koosearch_v1,239^v3^insert_chatgpt"}} ] [.reference_item] [ .reference_list ]
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值