深度解析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

相关文章

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

从基础到高级详解Go语言中错误处理的实践指南

《从基础到高级详解Go语言中错误处理的实践指南》Go语言采用了一种独特而明确的错误处理哲学,与其他主流编程语言形成鲜明对比,本文将为大家详细介绍Go语言中错误处理详细方法,希望对大家有所帮助... 目录1 Go 错误处理哲学与核心机制1.1 错误接口设计1.2 错误与异常的区别2 错误创建与检查2.1 基础

一文解析C#中的StringSplitOptions枚举

《一文解析C#中的StringSplitOptions枚举》StringSplitOptions是C#中的一个枚举类型,用于控制string.Split()方法分割字符串时的行为,核心作用是处理分割后... 目录C#的StringSplitOptions枚举1.StringSplitOptions枚举的常用

Python函数作用域与闭包举例深度解析

《Python函数作用域与闭包举例深度解析》Python函数的作用域规则和闭包是编程中的关键概念,它们决定了变量的访问和生命周期,:本文主要介绍Python函数作用域与闭包的相关资料,文中通过代码... 目录1. 基础作用域访问示例1:访问全局变量示例2:访问外层函数变量2. 闭包基础示例3:简单闭包示例4

MyBatis延迟加载与多级缓存全解析

《MyBatis延迟加载与多级缓存全解析》文章介绍MyBatis的延迟加载与多级缓存机制,延迟加载按需加载关联数据提升性能,一级缓存会话级默认开启,二级缓存工厂级支持跨会话共享,增删改操作会清空对应缓... 目录MyBATis延迟加载策略一对多示例一对多示例MyBatis框架的缓存一级缓存二级缓存MyBat

前端缓存策略的自解方案全解析

《前端缓存策略的自解方案全解析》缓存从来都是前端的一个痛点,很多前端搞不清楚缓存到底是何物,:本文主要介绍前端缓存的自解方案,文中通过代码介绍的非常详细,需要的朋友可以参考下... 目录一、为什么“清缓存”成了技术圈的梗二、先给缓存“把个脉”:浏览器到底缓存了谁?三、设计思路:把“发版”做成“自愈”四、代码

Java集合之Iterator迭代器实现代码解析

《Java集合之Iterator迭代器实现代码解析》迭代器Iterator是Java集合框架中的一个核心接口,位于java.util包下,它定义了一种标准的元素访问机制,为各种集合类型提供了一种统一的... 目录一、什么是Iterator二、Iterator的核心方法三、基本使用示例四、Iterator的工

Go语言中json操作的实现

《Go语言中json操作的实现》本文主要介绍了Go语言中的json操作的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧... 目录 一、jsOChina编程N 与 Go 类型对应关系️ 二、基本操作:编码与解码 三、结构体标签(Struc