[go]slice
Posted iiiiiher
tags:
篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了[go]slice相关的知识,希望对你有一定的参考价值。
切片的结构
切片是一个结构体
type slice struct {
array unsafe.Pointer //指向底层数组
len int
cap int
}
占用内存大小: 在64bit系统上占24个字节.
//word大小
32bit: 1word=4byte
64bit: 1word=8byte
//在64bit系统上slice占24个字节
type slice struct {
array unsafe.Pointer //8byte
len int //8byte
cap int //8byte
}
fmt.Println(unsafe.Sizeof([]int{})) //24bytes
切片数据包含在与切片关联的底层数组里, 拷贝切片时, 不会涉及底层数组数组的拷贝, 仅拷贝24bytes
定义
字面量方式
// 字面量方式定义
var arr[]int //nil切片: 指向nil
arr:= []int{} //空切片, 有内存地址分配
arr:=make([]int,0)
println(a==nil,b==nil) //只能和nil比较, slice引用类型之间不支持==运算
b:= []int{1,2,5:10}
make方式
// make方式:
make([]T, len, cap) //len参数必须, cap参数可选, 默认cap==size
len: 用于限定可读写的元素数量.
cap: 表示切片所引用数组片段的真实长度.
arr:=make([]int, 3, 5)
arr := make([]int,3)
fmt.Println(len(arr), cap(arr)) //3 3
arr := make([]int, 3, 4)
fmt.Printf("%#v",arr) //[]int{0, 0, 0}
- len和cap的说明
//len: 用于限定可读写的元素数量.
arr := make([]int, 2, 3)
arr[0] = 0
arr[1] = 1
arr[2] = 2 // index out of range [2] with length 2
如果要扩容slice, 使用append函数(making slice a dynamic data structure). 当append时:
- 如果cap够, append使用原数组
- 如果cap不够, append会新开辟一个backend array内存空间. 并重置cap. 当cap<1000时, 成倍的增长cap; 当cap>1000,增长因子为1.25.
从数组/切片reslice
// 从数组/切片获取
arr = [i,j,k]
len = j-i
cap = k-i
// 一个例子
x:= [...]int{0,1,2,3,4,5,6,7,8,9}
操作 得到的切片 len cap 备注
-------------+--------------------------+----+------+------------------------------
x[:] [0 1 2 3 4 5 6 7 8 9] 10 10 x[0:len(x)]
x[2:5] [2 3 4] 3 8
x[2:5:7] [2 3 4] 3 5
x[4:] [4 5 6 7 8 9] 6 6 x[4:len(x)]
x[:4] [0 1 2 3] 4 10 x[0:4]
x[:4:6] [0 1 2 3] 4 6 x[0:4:6]
使用 arr = [i,j], 注 这里的默认cap,是原始arr的长度-i
使用arr = [i,j,k]
使用arr = [i,j,k], 设置j=k, 即len=cap的好处: 修改切片内容时, 避免影响其他切片.
slice是引用类型
slice只能和nil比较
slice的赋值
切片是一个结构体, 切片在64bit系统下占用24bytes. 与切片关联的数据存放在底层数组里,不属于切片本身.
所以将切片赋值到任意变量时, 对底层数组的大小都不会有影响, 赋值时只会复制切片本身, 不会涉及底层数组里数据.
- slice函数参数传递
Go 语言的函数参数传递,只有值传递,没有引用传递。
函数形参是一个局部变量, 调用函数时, 会将变量拷贝一份, 赋值给函数形参.
这里slice的拷贝, 仅仅是拷贝slice的结构体, 不会涉及与slice关联的底层数组的数据.
// 函数foo接收一个整型切片,并返回这个切片
func foo(slice []int) []int {
...
return slice
}
func main(){
// 分配包含100万个整型值的切片
slice := make([]int, 1e6)
// 将slice传递到函数foo
slice = foo(slice)
}
以上是关于[go]slice的主要内容,如果未能解决你的问题,请参考以下文章
Go语言技巧之正确高效使用slice(听课笔记总结--简单易懂)