您的位置:首页 > 编程语言 > Go语言

Go语言学习笔记 --- slice切片

2017-08-07 23:39 726 查看
学习笔记根据 无闻 go语言基础教程 整理

slice切片

其本身并不是数组,它指向底层的数组

作为变长数组的替代方案,可以关联底层数组的局部或全部

类型为引用类型

可以直接创建或从底层数组获取生成

使用len() 获取元素个数, cap() 获取容量

一般使用make() 创建

如果多个slice指向相同底层数组,其中一个的值改变会影响全部

make([]T,len,cap)

其中cap可以省略,则和len的值相同

len表示存数的元素个数,cap表示容量

Reslice 时索引以被slice的切片为准

索引不可以超过被slice的切片的容量cap()值

索引月切不会导致底层数组的重新分配而是引发错误

可以在slice尾部追加元素

可以将一个slice追加在另一个slice尾部

如果最终长度未超过追加到slice的容量则返回原始slice

如果超过追加到的slice的容量则将重新分配数组并拷贝原始数据

示例代码:

package main

import (
"fmt"
)

func main() {
test1()
test2()
test3()
test4()
test5()
test6()
test7()
test8()
test9()
test10()
test11()
}

func test1() {
var s1 []int
fmt.Println(s1) // []

}

func test2() {
a := [5]int{1,2,3,4,5}
fmt.Println(a) // [1 2 3 4 5]
s1 := a[2]
fmt.Println(s1) // 3 此处是取值
s2 := a[4:5] // 左闭右开 此处是取数组
s3 := a[4:len(a)]
s4 := a[4:]
fmt.Println(s2) // [5]
fmt.Println(s3) // [5]
fmt.Println(s4) // [5]
// 备注 s2, s3, s4 都是从索引为4的位置,取值到最后
}

func test3() {
a := [5]int{1,2,3,4,5}
fmt.Println(a) // [1 2 3 4 5]
s1 := a[:3] // 取前3项,不会包含索引为3的,同样是左闭右开
fmt.Println(s1) // [1 2 3]
}

func test4() {
s1 := make([]int, 3, 5) // 初始化3个int放入数组,第三个参数是初始容量
fmt.Println(s1) // [0 0 0]
fmt.Println(len(s1),cap(s1)) // 3 5
}

func test5() {
a := []byte{'a','b','c','d','e','f','g','h','i','j'}
b := string(a[2:5])
c := string(a[3:5])
fmt.Println(b) // cde
fmt.Println(c) // de
}

func test6() {
s1 := make([]int, 3, 6)
fmt.Printf("%p\n", s1) // 0xc042047f50
s1 = append(s1,1,2,3)
fmt.Printf("%v %p\n", s1, s1) // [0 0 0 1 2 3] 0xc042047f50 // 此处表明,在未超过最大容量候,操作的都是同一个地址
s1 = append(s1,1,2,3)
fmt.Println("cap:", cap(s1)) // cap:12 // 此处可以看到原容量是6,现在就会扩大一倍,每次都是一倍一倍的增长,地址也是重新分配
fmt.Printf("%v %p\n", s1, s1) // [0 0 0 1 2 3 1 2 3] 0xc0420420c0 // 此处地址变化了
}

func test7() {
a := []int{1,2,3,4,5}
s1 := a[2:5]
s2 := a[1:3]
fmt.Println(s1, s2) // [3 4 5] [2 3] // 可以看出 s1 和 s2 底层共享元素3的,指向的是底层数组的同一个元素
s1[0] = 9 // 改变这个元素 在s1中是索引为0的数是3,现在变成9
fmt.Println(s1, s2) // [9 4 5] [2 9]
}

func test8() {
a := []int{1,2,3,4,5}
s1 := a[2:5]
s2 := a[1:3]
fmt.Println(s1, s2) // [3 4 5] [2 3] // 可以看出 s1 和 s2 底层共享元素3的,指向的是底层数组的同一个元素
fmt.Println(len(s2),cap(s2)) // 2 4
s2 = append(s2, 1,1,1) // append 超越s2的容量, 新的内存地址被分配,将不指向同一个底层数组了, 此时就不会被改变
s1[0] = 9 // 改变这个元素 在s1中是索引为0的数是3,现在变成9
fmt.Println(s1, s2) // [9 4 5] [2 3 1 1 1]
}

func test9() {
s1 := []int{1,2,3,4,5,6}
s2 := []int{7,8,9}
copy(s1,s2) // 将s2拷贝到s1中
fmt.Println(s1) // [7 8 9 4 5 6] // 可以看出保持s1的长度
}

func test10() {
s1 := []int{1,2,3,4,5,6}
s2 := []int{7,8,9}
copy(s2,s1) // 将s1拷贝到s2中
fmt.Println(s2) // [1 2 3] // 可以看出保持上s2的长度
}

func test11() {
s1 := []int{1,2,3,4,5,6}
s2 := []int {7,8,9,10,1,1,1,1,1}
copy(s2[2:4], s1[1:3])
fmt.Println(s2) // [7 8 2 3 1 1 1 1 1]
}
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: