深度解析Go语言中的Slice切片

2024-06-03 05:20

本文主要是介绍深度解析Go语言中的Slice切片,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

深度解析Go语言中的Slice切片

  • 一、 简介
  • 二、数据结构
  • 三、初始化
  • 四、内容截取
  • 五、切片扩容
  • 六、元素删除
  • 七、切片拷贝


一、 简介

go中的切片,在某种程度上相当于别的语言中的“数组”。不同点在于切片的长度和容量是可变的,在使用过程中可以进行扩容。

二、数据结构

type slice struct {array unsafe.Pointerlen   intcap   int
}

这就是切片定义的底层源代码,非常简洁

array :指向切片引用的底层数组,由Go运行时使用unsafe.Pointer管理,允许切片中的任何类型元素。

len:这是切片的长度,代表它包含的元素数量。

cap:这是切片的容量,即在需要分配新的底层数组之前,切片可以容纳的元素的最大数量。

由此我们不难发现,切片内部如果储存数据,还是靠指向底层数组的指针实现的,所以,如果传递切片,那么进行的就是引用传递操作了

三、初始化

初始化可以有以下形式

	// 声明但不初始化var a []int// 基于 make 进行初始化 len = cap = 10b := make([]int, 10)// 基于 make 进行初始化 len = 10 cap = 20c := make([]int, 10, 20)// 直接赋值 len = cap = 10d := []int{1,2, 3, 4, 5, 6, 7, 8, 9, 10}

PS:

  1. cap 必须大于len,否则会报错
  2. 如果len<cap,则访问超出len的元素会报错——数组越界
  3. 指定长度但是并未赋值,此时数组长度内的元素全部为该类型的零值
  4. 只定义但未声明时,此时变量为空指针nil

源代码:

func makeslice(et *_type, len, cap int) unsafe.Pointer {mem, overflow := math.MulUintptr(et.Size_, uintptr(cap))if overflow || mem > maxAlloc || len < 0 || len > cap {// 注意:当有人使用make([]T, bignumber)时,产生'len超出范围'的错误,// 而不是'cap超出范围'的错误。'cap超出范围'也是对的,但由于cap只是隐式提供的,// 所以说len更清楚。// 参见 golang.org/issue/4085。mem, overflow := math.MulUintptr(et.Size_, uintptr(len))if overflow || mem > maxAlloc || len < 0 {panicmakeslicelen()}panicmakeslicecap()}return mallocgc(mem, et, true)
}

解释:

  • 用来计算所需内存的大小
mem, overflow := math.MulUintptr(et.Size_, uintptr(cap))
  • 检查是否有溢出、内存超限或无效的长度和容量
if overflow || mem > maxAlloc || len < 0 || len > cap
  • 内存超限就直接抛出错误
  • 调用mallocgc方法进行内存分配

四、内容截取

可以使用下面这种方式对切片进行内容截取

	s := []int{1, 2, 3, 4, 5, 6, 7, 8, 9}// s1: [2 3 4 5 6 7 8 9]s1 := s[1:]// s2: [1 2 3 4 5 6 7 8]s2 := s[:len(s)-1]// s3: [2 3 4 5 6 7 8]s3 := s[1 : len(s)-1]

PS:其实不管进行什么截取操作,本质上都没有创造新的数组,底层的数组仍然是初始的那一个没有变,只是改变了起始指针的位置,len以及cap的值。

五、切片扩容

func growslice(oldPtr unsafe.Pointer, newLen, oldCap, num int, et *_type) slice {oldLen := newLen - num// 如果启用了竞态检测,则进行内存读取范围检测if raceenabled {callerpc := getcallerpc()racereadrangepc(oldPtr, uintptr(oldLen*int(et.Size_)), callerpc, abi.FuncPCABIInternal(growslice))}// 如果启用了内存清理检测,则进行内存读取检测if msanenabled {msanread(oldPtr, uintptr(oldLen*int(et.Size_)))}// 如果启用了地址清理检测,则进行内存读取检测if asanenabled {asanread(oldPtr, uintptr(oldLen*int(et.Size_)))}// 如果新长度小于0,则抛出异常if newLen < 0 {panic(errorString("growslice: len out of range"))}// 如果元素类型的大小为0,则返回一个新的切片,其指针为nil,长度和容量为newLenif et.Size_ == 0 {return slice{unsafe.Pointer(&zerobase), newLen, newLen}}// 计算新的容量newcap := oldCapdoublecap := newcap + newcapif newLen > doublecap {newcap = newLen} else {const threshold = 256if oldCap < threshold {newcap = doublecap} else {for 0 < newcap && newcap < newLen {newcap += (newcap + 3*threshold) / 4}if newcap <= 0 {newcap = newLen}}}// 根据元素类型的大小,计算内存大小,并检查是否溢出var overflow boolvar lenmem, newlenmem, capmem uintptrswitch {case et.Size_ == 1:lenmem = uintptr(oldLen)newlenmem = uintptr(newLen)capmem = roundupsize(uintptr(newcap))overflow = uintptr(newcap) > maxAllocnewcap = int(capmem)case et.Size_ == goarch.PtrSize:lenmem = uintptr(oldLen) * goarch.PtrSizenewlenmem = uintptr(newLen) * goarch.PtrSizecapmem = roundupsize(uintptr(newcap) * goarch.PtrSize)overflow = uintptr(newcap) > maxAlloc/goarch.PtrSizenewcap = int(capmem / goarch.PtrSize)case isPowerOfTwo(et.Size_):var shift uintptrif goarch.PtrSize == 8 {shift = uintptr(sys.TrailingZeros64(uint64(et.Size_))) & 63} else {shift = uintptr(sys.TrailingZeros32(uint32(et.Size_))) & 31}lenmem = uintptr(oldLen) << shiftnewlenmem = uintptr(newLen) << shiftcapmem = roundupsize(uintptr(newcap) << shift)overflow = uintptr(newcap) > (maxAlloc >> shift)newcap = int(capmem >> shift)default:lenmem = uintptr(oldLen) * et.Size_newlenmem = uintptr(newLen) * et.Size_capmem, overflow = math.MulUintptr(et.Size_, uintptr(newcap))capmem = roundupsize(capmem)newcap = int(capmem / et.Size_)}// 检查是否溢出,以防止在32位架构上触发段错误if overflow || capmem > maxAlloc {panic(errorString("growslice: len out of range"))}// 分配内存,并根据情况清理内存var p unsafe.Pointerif et.PtrBytes == 0 {p = mallocgc(capmem, nil, false)memclrNoHeapPointers(add(p, newlenmem), capmem-newlenmem)} else {p = mallocgc(capmem, et, true)if lenmem > 0 && writeBarrier.enabled {bulkBarrierPreWriteSrcOnly(uintptr(p), uintptr(oldPtr), lenmem-et.Size_+et.PtrBytes)}}// 将旧切片的数据移动到新的内存位置memmove(p, oldPtr, lenmem)// 返回新的切片return slice{p, newLen, newcap}
}

主要包含以下内容:

  • 检查新长度是否合法,如果不合法则抛出异常。
  • 计算新的容量,如果新长度超过当前容量的两倍,则直接使用新长度作为新容量;否则,根据一定的规则逐步增加容量,直到满足需求。
  • 分配新的内存空间,并将旧切片的数据复制到新的内存空间。
  • 返回一个新的切片,其底层数组指向新分配的内存,长度和容量更新为新的值。
    PS :
    倘若老容量小于 256,则直接采用老容量的2倍作为新容量;倘若老容量已经大于等于 256,则在老容量的基础上扩容 1/4 的比例并且累加上 192 的数值,持续这样处理,直到得到的新容量已经大于等于预期的新容量为止

六、元素删除

删除其实本质上跟截取是一样的

	s := []int{0, 1, 2, 3, 4}// [1,2,3,4]s = s[1:]
	s := []int{0, 1, 2, 3, 4}// [0,1,2,3]s = s[0 : len(s)-1]

七、切片拷贝

切片拷贝有两种方式
一种是普通的简单拷贝,就是引用传递

s := []int{0, 1, 2, 3, 4}
s1 := s

另一种是深度拷贝,创建出一个和 slice 容量大小相等的独立的内存区域,并将原 slice 中的元素一一拷贝到新空间中

s := []int{0, 1, 2, 3, 4}
s1 := make([]int, len(s))
copy(s1, s)

这篇关于深度解析Go语言中的Slice切片的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1026106

相关文章

python使用fastapi实现多语言国际化的操作指南

《python使用fastapi实现多语言国际化的操作指南》本文介绍了使用Python和FastAPI实现多语言国际化的操作指南,包括多语言架构技术栈、翻译管理、前端本地化、语言切换机制以及常见陷阱和... 目录多语言国际化实现指南项目多语言架构技术栈目录结构翻译工作流1. 翻译数据存储2. 翻译生成脚本

Go路由注册方法详解

《Go路由注册方法详解》Go语言中,http.NewServeMux()和http.HandleFunc()是两种不同的路由注册方式,前者创建独立的ServeMux实例,适合模块化和分层路由,灵活性高... 目录Go路由注册方法1. 路由注册的方式2. 路由器的独立性3. 灵活性4. 启动服务器的方式5.

Java深度学习库DJL实现Python的NumPy方式

《Java深度学习库DJL实现Python的NumPy方式》本文介绍了DJL库的背景和基本功能,包括NDArray的创建、数学运算、数据获取和设置等,同时,还展示了如何使用NDArray进行数据预处理... 目录1 NDArray 的背景介绍1.1 架构2 JavaDJL使用2.1 安装DJL2.2 基本操

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动

Go语言中三种容器类型的数据结构详解

《Go语言中三种容器类型的数据结构详解》在Go语言中,有三种主要的容器类型用于存储和操作集合数据:本文主要介绍三者的使用与区别,感兴趣的小伙伴可以跟随小编一起学习一下... 目录基本概念1. 数组(Array)2. 切片(Slice)3. 映射(Map)对比总结注意事项基本概念在 Go 语言中,有三种主要

Go Mongox轻松实现MongoDB的时间字段自动填充

《GoMongox轻松实现MongoDB的时间字段自动填充》这篇文章主要为大家详细介绍了Go语言如何使用mongox库,在插入和更新数据时自动填充时间字段,从而提升开发效率并减少重复代码,需要的可以... 目录前言时间字段填充规则Mongox 的安装使用 Mongox 进行插入操作使用 Mongox 进行更

C语言中自动与强制转换全解析

《C语言中自动与强制转换全解析》在编写C程序时,类型转换是确保数据正确性和一致性的关键环节,无论是隐式转换还是显式转换,都各有特点和应用场景,本文将详细探讨C语言中的类型转换机制,帮助您更好地理解并在... 目录类型转换的重要性自动类型转换(隐式转换)强制类型转换(显式转换)常见错误与注意事项总结与建议类型

MySQL 缓存机制与架构解析(最新推荐)

《MySQL缓存机制与架构解析(最新推荐)》本文详细介绍了MySQL的缓存机制和整体架构,包括一级缓存(InnoDBBufferPool)和二级缓存(QueryCache),文章还探讨了SQL... 目录一、mysql缓存机制概述二、MySQL整体架构三、SQL查询执行全流程四、MySQL 8.0为何移除查

在Rust中要用Struct和Enum组织数据的原因解析

《在Rust中要用Struct和Enum组织数据的原因解析》在Rust中,Struct和Enum是组织数据的核心工具,Struct用于将相关字段封装为单一实体,便于管理和扩展,Enum用于明确定义所有... 目录为什么在Rust中要用Struct和Enum组织数据?一、使用struct组织数据:将相关字段绑

使用Java实现一个解析CURL脚本小工具

《使用Java实现一个解析CURL脚本小工具》文章介绍了如何使用Java实现一个解析CURL脚本的工具,该工具可以将CURL脚本中的Header解析为KVMap结构,获取URL路径、请求类型,解析UR... 目录使用示例实现原理具体实现CurlParserUtilCurlEntityICurlHandler