Go 其二 数组与切片
艺多不压身,学习一下最近蛮火的Go语言,整理一下笔记。相关Code和笔记也放到了Git上,传送门。
数组的声明
var a [3] int //声明并初始化为默认零值
a[0] = 1
b := [3]int{1,2,3} //声明同时初始化
c := [2][2]int{{1,2},{3,4}} //多维数组初始化
Go语言中定义的变量一定要使用,否则会报错,但如果只是想使用foreach的写法,可以使用 _ 来占位.
//foreach的写法, idx即为数组的索引的值
for idx, e:= range arr2 {
t.Log(idx, e)
}
//Go语言中定义的变量一定要使用,所以下面的写法会报错 idx declared but not used
// for idx, e:= range arr2 {
// t.Log(e)
// }
//如果只是想使用foreach的写法,可以使用 _ 来占位
for _, e:= range arr2 {
t.Log(e)
}
数组截取
a[开始索引(包含), 结束索引(不包含)]
a := [...] int {1,2,3,4,5}
a[1:2] // 2
a[1:3] // 2,3
a[1:len(1)] // 2, 3, 4, 5
a[1:] //2, 3, 4, 5 //从索引1开始一直到末尾
a[:3] //1,2,3 //从索引0开始,一直到索引为3的位置,不包含索引为3的值
---------------------------------------------------------------
切片
切片在别的语言中不一定有,比如c#
使用起来像是可变长数组
切片内部结构
实际上是一个结构体,包含三个元素
1.指针,指向一片连续的存储空间(一个数组)ptr
2.切片内的元素个数 len
3.指针指向的这个数组的长度,容量 cap
//与声明数组很类似,区别是没有指定长度!例如
var s0 []int
利用len()方法来获得切片内元素个数
利用cap()方法来获得切片空间大小
利用append(s0, 1)来填充切片
使用make方法来声明长度和容量不同的切片 例如
//这里s2长度为3,但容量为5
s2 := make([]int, 3, 5)
其中len个元素会被初始化为默认值0,未初始化元素不可访问
切片本质上是共享的存储结构,注意这里的共享指的是由同一个切片“派生”出的其他切片或数组,例如
year := []string{"Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"}
q2 := year[3:6]
summer := year[5:8]
year,q2,summer是共享存储空间的
而反之,如
a := []int{1,2,3,4}
b := []int{1,2,3,4}
虽然初始的值相同,但其实是不共享存储空间的
---------------------------------------------------------------
数组 VS 切片
1. 容量是否可以伸缩,数组不可以,切片可以
2. 是否可以进行比较,数组可以与数组比较,切片与切片之间不可以比较
附录:
数组相关Code:
package array_test import "testing" func TestArrayInit(t *testing.T){ var arr [3]int t.Log(arr[1],arr[2]) arr1 := [4]int{1,2,3,4} //不指定长度且初始化时可以用[...] arr2 := [...]int{1,2,3,4,5} arr1[1] = 9 t.Log(arr1[1], arr1[3]) t.Log(arr2[2]) } func TestArrayTraval(t *testing.T){ arr2 := [...]int{1,2,3,4,5} for i:= 0; i < len(arr2); i++{ t.Log(arr2[i]) } //foreach的写法, idx即为数组的索引的值 for idx, e:= range arr2 { t.Log(idx, e) } //Go语言中定义的变量一定要使用,所以下面的写法会报错 idx declared but not used // for idx, e:= range arr2 { // t.Log(e) // } //如果只是想使用foreach的写法,可以使用 _ 来占位 for _, e:= range arr2 { t.Log(e) } } func TestArraySection(t *testing.T){ arr3 := [...] int{1,2,3,4,5} arr3_sec := arr3[:3] t.Log(arr3_sec) arr3_sec2 := arr3[3:] t.Log(arr3_sec2) }
切片相关code:
package slice_test import "testing" func TestSliceInit(t *testing.T){ //与声明数组很类似,区别是没有指定长度! var s0 []int t.Log(len(s0), cap(s0)) s0 = append(s0, 1) t.Log(len(s0), cap(s0)) //初始化 s1 := []int{1,2,3,4} t.Log(len(s1), cap(s1)) //这里s2长度为3,但容量为5 s2 := make([]int, 3, 5) t.Log(len(s2), cap(s2)) //这里会报错,因为后两个元素会越界 //t.Log(s2[0],s2[1],s2[2],s2[3],s2[4]) t.Log(s2[0],s2[1],s2[2]) s2 = append(s2, 1) t.Log(s2[0],s2[1],s2[2],s2[3]) } func TestSliceGrowing(t *testing.T){ s:=[]int{} for i:=0; i<10; i++{ s = append(s, i) t.Log(len(s), cap(s)) } //输出结果 /* TestSliceGrowing: slice_test.go:30: 1 1 TestSliceGrowing: slice_test.go:30: 2 2 TestSliceGrowing: slice_test.go:30: 3 4 TestSliceGrowing: slice_test.go:30: 4 4 TestSliceGrowing: slice_test.go:30: 5 8 TestSliceGrowing: slice_test.go:30: 6 8 TestSliceGrowing: slice_test.go:30: 7 8 TestSliceGrowing: slice_test.go:30: 8 8 TestSliceGrowing: slice_test.go:30: 9 16 TestSliceGrowing: slice_test.go:30: 10 16 */ //每次不够放的适合,都会将上次的扩展为之前的两倍,因此要使用类似于s2 = append(s2, 1) 因为是新的空间给s2,并将原来的数据拷贝过去。 } func TestSliceShareMemory(t *testing.T) { year := []string{"Jan", "Feb", "Mar", "Apr", "May", "Jun", "Jul", "Aug", "Sep", "Oct", "Nov", "Dec"} q2 := year[3:6] t.Log(q2, len(q2), cap(q2)) //输出结果为 //TestSliceShareMemory: slice_test.go:53: [Apr May Jun] 3 9,容量是直接到末尾的,所以是9. summer := year[5:8] t.Log(summer, len(summer), cap(summer)) summer[0] = "Unkonw" t.Log(summer, len(summer), cap(summer)) t.Log(q2, len(q2), cap(q2)) //输出结果为 /* TestSliceShareMemory: slice_test.go:60: [Unkonw Jul Aug] 3 7 TestSliceShareMemory: slice_test.go:61: [Apr May Unkonw] 3 9 */ //由于是共享的存储空间,因此summer的改动同时影响到了q2的值. } func TestSliceComparing(t *testing.T){ a := []int{1,2,3,4} b := []int{1,2,3,4} //会报错,nvalid operation: a == b (slice can only be compared to nil) // if(a == b){ // t.Log("Can Compare") // } t.Log(a[2]) t.Log(b[2]) b[2] = 9 t.Log(a[2]) t.Log(b[2]) }