[译]Go Slice 秘籍
smallnest · · 1434 次点击 · · 开始浏览这是 Golang官方的一个总结: SliceTricks
由于引入了内建的append的方法, 包container/vector的很多方法都被移除了,可以被内建的append和copy方法代替。
下面是栈vector的操作方法的实现,使用slice实现相关的操作。
AppendVector
1
a = append(a, b...)
Copy
1234
b = make([]T, len(a))copy(b, a)// 如果a不为空,也可以用下面的方式b = append([]T(nil), a...)
Cut
切掉一段数据
1
a = append(a[:i], a[j:]...)
Delete
删除一个元素
123
a = append(a[:i], a[i+1:]...)// ora = a[:i+copy(a[i:], a[i+1:])]
Delete,但不保持原来顺序
12
a[i] = a[len(a)-1]a = a[:len(a)-1]
注意:如果元素是一个指针,或者是一个包含指针字段的struct,上面的cut、delete实现可能会有潜在的内存泄漏的问题。一些元素的值可能会被a一直引用而不被释放,下面的代码可以解决这个问题。
Cut
12345
copy(a[i:], a[j:])for k, n := len(a)-j+i, len(a); k < n; k++ {a[k] = nil // or the zero value of T}a = a[:len(a)-j+i]
Delete
123
copy(a[i:], a[i+1:])a[len(a)-1] = nil // or the zero value of Ta = a[:len(a)-1]
Delete,但不保持原来顺序
123
a[i] = a[len(a)-1]a[len(a)-1] = nila = a[:len(a)-1]
Expand
插入一段到中间
1
a = append(a[:i], append(make([]T, j), a[i:]...)...)
Extend
插入一段到尾部
1
a = append(a, make([]T, j)...)
Insert
1
a = append(a[:i], append([]T{x}, a[i:]...)...)
注意: 第二个append会使用它底层的存储创建一个新的slice,然后复制a[i:]到这个slice,然后把这个slice再复制回s。 新的slice的创建和第二次copy可以使用下面的方式来避免:
Insert
123
s = append(s, 0)copy(s[i+1:], s[i:])s[i] = x
InsertVector
1
a = append(a[:i], append(b, a[i:]...)...)
Pop
1
x, a = a[len(a)-1], a[:len(a)-1]
Push
1
a = append(a, x)
Shift
1
x, a := a[0], a[1:]
Unshift
1
a = append([]T{x}, a...)
其它技巧
无额外对象分配的filter
这个技巧利用了slice会共享它的底层的数据存储和容量。
123456
b := a[:0]for _, x := range a {if f(x) {b = append(b, x)}}
反转
将slice中的元素反转。
1234
for i := len(a)/2-1; i >= 0; i-- {opp := len(a)-1-ia[i], a[opp] = a[opp], a[i]}
下面的代码类似,只不过使用了两个索引变量
123
for left, right := 0, len(a)-1; left < right; left, right = left+1, right-1 {a[left], a[right] = a[right], a[left]}
有疑问加站长微信联系(非本文作者)
入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889
关注微信- 请尽量让自己的回复能够对别人有帮助
- 支持 Markdown 格式, **粗体**、~~删除线~~、
`单行代码` - 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
- 图片支持拖拽、截图粘贴等方式上传
收入到我管理的专栏 新建专栏
这是 Golang官方的一个总结: SliceTricks
由于引入了内建的append的方法, 包container/vector的很多方法都被移除了,可以被内建的append和copy方法代替。
下面是栈vector的操作方法的实现,使用slice实现相关的操作。
AppendVector
1
a = append(a, b...)
Copy
1234
b = make([]T, len(a))copy(b, a)// 如果a不为空,也可以用下面的方式b = append([]T(nil), a...)
Cut
切掉一段数据
1
a = append(a[:i], a[j:]...)
Delete
删除一个元素
123
a = append(a[:i], a[i+1:]...)// ora = a[:i+copy(a[i:], a[i+1:])]
Delete,但不保持原来顺序
12
a[i] = a[len(a)-1]a = a[:len(a)-1]
注意:如果元素是一个指针,或者是一个包含指针字段的struct,上面的cut、delete实现可能会有潜在的内存泄漏的问题。一些元素的值可能会被a一直引用而不被释放,下面的代码可以解决这个问题。
Cut
12345
copy(a[i:], a[j:])for k, n := len(a)-j+i, len(a); k < n; k++ {a[k] = nil // or the zero value of T}a = a[:len(a)-j+i]
Delete
123
copy(a[i:], a[i+1:])a[len(a)-1] = nil // or the zero value of Ta = a[:len(a)-1]
Delete,但不保持原来顺序
123
a[i] = a[len(a)-1]a[len(a)-1] = nila = a[:len(a)-1]
Expand
插入一段到中间
1
a = append(a[:i], append(make([]T, j), a[i:]...)...)
Extend
插入一段到尾部
1
a = append(a, make([]T, j)...)
Insert
1
a = append(a[:i], append([]T{x}, a[i:]...)...)
注意: 第二个append会使用它底层的存储创建一个新的slice,然后复制a[i:]到这个slice,然后把这个slice再复制回s。 新的slice的创建和第二次copy可以使用下面的方式来避免:
Insert
123
s = append(s, 0)copy(s[i+1:], s[i:])s[i] = x
InsertVector
1
a = append(a[:i], append(b, a[i:]...)...)
Pop
1
x, a = a[len(a)-1], a[:len(a)-1]
Push
1
a = append(a, x)
Shift
1
x, a := a[0], a[1:]
Unshift
1
a = append([]T{x}, a...)
其它技巧
无额外对象分配的filter
这个技巧利用了slice会共享它的底层的数据存储和容量。
123456
b := a[:0]for _, x := range a {if f(x) {b = append(b, x)}}
反转
将slice中的元素反转。
1234
for i := len(a)/2-1; i >= 0; i-- {opp := len(a)-1-ia[i], a[opp] = a[opp], a[i]}
下面的代码类似,只不过使用了两个索引变量
123
for left, right := 0, len(a)-1; left < right; left, right = left+1, right-1 {a[left], a[right] = a[right], a[left]}