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

相关文章

SpringCloud动态配置注解@RefreshScope与@Component的深度解析

《SpringCloud动态配置注解@RefreshScope与@Component的深度解析》在现代微服务架构中,动态配置管理是一个关键需求,本文将为大家介绍SpringCloud中相关的注解@Re... 目录引言1. @RefreshScope 的作用与原理1.1 什么是 @RefreshScope1.

Java并发编程必备之Synchronized关键字深入解析

《Java并发编程必备之Synchronized关键字深入解析》本文我们深入探索了Java中的Synchronized关键字,包括其互斥性和可重入性的特性,文章详细介绍了Synchronized的三种... 目录一、前言二、Synchronized关键字2.1 Synchronized的特性1. 互斥2.

go中空接口的具体使用

《go中空接口的具体使用》空接口是一种特殊的接口类型,它不包含任何方法,本文主要介绍了go中空接口的具体使用,具有一定的参考价值,感兴趣的可以了解一下... 目录接口-空接口1. 什么是空接口?2. 如何使用空接口?第一,第二,第三,3. 空接口几个要注意的坑坑1:坑2:坑3:接口-空接口1. 什么是空接

Java的IO模型、Netty原理解析

《Java的IO模型、Netty原理解析》Java的I/O是以流的方式进行数据输入输出的,Java的类库涉及很多领域的IO内容:标准的输入输出,文件的操作、网络上的数据传输流、字符串流、对象流等,这篇... 目录1.什么是IO2.同步与异步、阻塞与非阻塞3.三种IO模型BIO(blocking I/O)NI

Python 中的异步与同步深度解析(实践记录)

《Python中的异步与同步深度解析(实践记录)》在Python编程世界里,异步和同步的概念是理解程序执行流程和性能优化的关键,这篇文章将带你深入了解它们的差异,以及阻塞和非阻塞的特性,同时通过实际... 目录python中的异步与同步:深度解析与实践异步与同步的定义异步同步阻塞与非阻塞的概念阻塞非阻塞同步

C语言中的数据类型强制转换

《C语言中的数据类型强制转换》:本文主要介绍C语言中的数据类型强制转换方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录C语言数据类型强制转换自动转换强制转换类型总结C语言数据类型强制转换强制类型转换:是通过类型转换运算来实现的,主要的数据类型转换分为自动转换

利用Go语言开发文件操作工具轻松处理所有文件

《利用Go语言开发文件操作工具轻松处理所有文件》在后端开发中,文件操作是一个非常常见但又容易出错的场景,本文小编要向大家介绍一个强大的Go语言文件操作工具库,它能帮你轻松处理各种文件操作场景... 目录为什么需要这个工具?核心功能详解1. 文件/目录存javascript在性检查2. 批量创建目录3. 文件

C语言实现两个变量值交换的三种方式

《C语言实现两个变量值交换的三种方式》两个变量值的交换是编程中最常见的问题之一,以下将介绍三种变量的交换方式,其中第一种方式是最常用也是最实用的,后两种方式一般只在特殊限制下使用,需要的朋友可以参考下... 目录1.使用临时变量(推荐)2.相加和相减的方式(值较大时可能丢失数据)3.按位异或运算1.使用临时

使用C语言实现交换整数的奇数位和偶数位

《使用C语言实现交换整数的奇数位和偶数位》在C语言中,要交换一个整数的二进制位中的奇数位和偶数位,重点需要理解位操作,当我们谈论二进制位的奇数位和偶数位时,我们是指从右到左数的位置,本文给大家介绍了使... 目录一、问题描述二、解决思路三、函数实现四、宏实现五、总结一、问题描述使用C语言代码实现:将一个整

Redis中高并发读写性能的深度解析与优化

《Redis中高并发读写性能的深度解析与优化》Redis作为一款高性能的内存数据库,广泛应用于缓存、消息队列、实时统计等场景,本文将深入探讨Redis的读写并发能力,感兴趣的小伙伴可以了解下... 目录引言一、Redis 并发能力概述1.1 Redis 的读写性能1.2 影响 Redis 并发能力的因素二、