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])

	
}

 

posted @ 2020-06-16 22:23  DogTwo  阅读(46)  评论(0编辑  收藏