赞
踩
go中的切片,在某种程度上相当于别的语言中的“数组”。不同点在于切片的长度和容量是可变的,在使用过程中可以进行扩容。
type slice struct {
array unsafe.Pointer
len int
cap int
}
这就是切片定义的底层源代码,非常简洁
array :指向切片引用的底层数组,由Go运行时使用unsafe.Pointer管理,允许切片中的任何类型元素。
len:这是切片的长度,代表它包含的元素数量。
cap:这是切片的容量,即在需要分配新的底层数组之前,切片可以容纳的元素的最大数量。
由此我们不难发现,切片内部如果储存数据,还是靠指向底层数组的指针实现的,所以,如果传递切片,那么进行的就是引用传递操作了
初始化可以有以下形式
// 声明但不初始化
var a []int
// 基于 make 进行初始化 len = cap = 10
b := make([]int, 10)
// 基于 make 进行初始化 len = 10 cap = 20
c := make([]int, 10, 20)
// 直接赋值 len = cap = 10
d := []int{1,2, 3, 4, 5, 6, 7, 8, 9, 10}
PS:
源代码:
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
可以使用下面这种方式对切片进行内容截取
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,长度和容量为newLen if et.Size_ == 0 { return slice{unsafe.Pointer(&zerobase), newLen, newLen} } // 计算新的容量 newcap := oldCap doublecap := newcap + newcap if newLen > doublecap { newcap = newLen } else { const threshold = 256 if oldCap < threshold { newcap = doublecap } else { for 0 < newcap && newcap < newLen { newcap += (newcap + 3*threshold) / 4 } if newcap <= 0 { newcap = newLen } } } // 根据元素类型的大小,计算内存大小,并检查是否溢出 var overflow bool var lenmem, newlenmem, capmem uintptr switch { case et.Size_ == 1: lenmem = uintptr(oldLen) newlenmem = uintptr(newLen) capmem = roundupsize(uintptr(newcap)) overflow = uintptr(newcap) > maxAlloc newcap = int(capmem) case et.Size_ == goarch.PtrSize: lenmem = uintptr(oldLen) * goarch.PtrSize newlenmem = uintptr(newLen) * goarch.PtrSize capmem = roundupsize(uintptr(newcap) * goarch.PtrSize) overflow = uintptr(newcap) > maxAlloc/goarch.PtrSize newcap = int(capmem / goarch.PtrSize) case isPowerOfTwo(et.Size_): var shift uintptr if goarch.PtrSize == 8 { shift = uintptr(sys.TrailingZeros64(uint64(et.Size_))) & 63 } else { shift = uintptr(sys.TrailingZeros32(uint32(et.Size_))) & 31 } lenmem = uintptr(oldLen) << shift newlenmem = uintptr(newLen) << shift capmem = 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.Pointer if 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} }
主要包含以下内容:
删除其实本质上跟截取是一样的
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)
Copyright © 2003-2013 www.wpsshop.cn 版权所有,并保留所有权利。