11.深入理解切片(slice)

.container .card .information strong · · 259 次点击 · · 开始浏览    
这是一个创建于 的文章,其中的信息可能已经有所发展或是发生改变。

本文视频地址

slice是 Go 语言在数组之上提供的一个重要的抽象数据类型。在绝大多数需要使用数组的场合,切片都实现了完美替代。并且和数组相比,切片提供了更通用、功能更强大且便捷的数据序列访问接口。

1.什么是数组

Go 语言数组是一个固定长度的、容纳同构类型元素的连续序列。因此 Go 数组类型具有两个属性:元素类型和数组长度。但凡这两个属性相同的数组类型是等价的。比如下面变量 a、b、c 对应的数组类型是三个不同的数组类型:

var a [8]string
var b [8]byte
var c [9]string

变量 a、b 对应的数组类型长度属性相同,但元素类型不同,a是string,b是byte.
变量 a、c 对应的数组类型的元素类型相同,都是 int,但数组类型的长度不(a是 8,c是 9)

Go 的数组是值类型,这点和JAVA完全不同。在JAVA中数组是引用类型。在Go语言中,传递数组是纯粹的值拷贝。

2.切片究竟是什么

在 Go 语言中,数组更多是底层存储空间的角色;而切片是为底层的数组打开了一个访问的“窗口”。下面看一下切片的源代码

//$GOROOT/src/runtime/slice.go

type slice struct {
        array unsafe.Pointer
        len   int
        cap   int
}
array:是指向下层数组某元素的指针,该元素也是切片的起始元素;
len:是切片的长度,即切片中当前元素的个数;
cap:是切片的最大容量,cap >= len;

在运行时中,每个切片变量都是一个 runtime.slice 结构体的实例。
创建一个 slice 实例:s1 := make([]byte, 5),编译器会自动为切片建立一个底层数组,如果没有在 make 中指定 cap 参数,那么 cap = len。

我们还可以创建对已存在数组进行操作的切片,语法 u[low:max] :
u := [10]byte{10, 12, 23, 14, 15, 16, 17, 88, 69, 20}
s := u[3:7]
切片是 [14 15 16 17]
通过 s 看到的第一个元素是 u[3],我们通过 s 能看到并操作的数组元素是 4 个(max-low)。切片的 cap 值取决于底层数组的长度。我们看到从切片 s 的第一个元素 s[0],即 u[3] 到数组末尾一共有 7 个元素,因此切片 s 的 cap 为 7。

当切片作为函数参数传递给函数时,实际传递的就是切片的内部表示,也就是上面的 runtime.slice 结构体实例,因此无论切片“描述”的底层数组有多大,切片作为参数传递带来的性能损耗都是小到可以忽略不计的。这就是为什么函数在参数中多使用切片而不用数组指针的原因之一。

3. 切片的高级特性:动态扩容

var s []int // s被赋予零值nil
s = append(s, 1) 

由于初值为零值,s 这个“描述符”并没有绑定对应的底层数组。而经过 append 操作后,s 显然已经“绑定”了属于它的底层数组。

append 会根据 slice 对底层数组容量的需求对底层数组进行动态调整。 append在当前底层数组容量无法满足的情况下,动态分配新的数组,新数组长度会按一定规律扩展(这里针对元素是 int 型的数组,新数组的容量为当前数组的 2 倍。其他类型的扩展系数可能有所不同),新数组建立后,append 会把旧数组中的数据 copy 到新数组中,之后新数组便成为了 slice 的底层数组,旧数组会被垃圾回收掉。

4. 尽量使用 cap 参数创建 slice

s := make([]T, 0, cap)
这样会避免扩容带来的性能开销,如果可以预估出切片底层数组需要承载的元素数量,强烈建议在创建 slice 时带上 cap 参数。

image


有疑问加站长微信联系(非本文作者)

本文来自:Segmentfault

感谢作者:.container .card .information strong

查看原文:11.深入理解切片(slice)

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

259 次点击  
加入收藏 微博
上一篇:10.零值可用
下一篇:12.Go 字符串
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传