为什么会有切片?

golang中数组的特点(缺点)
1、数组的长度在定义之后无法再次修改;
2、数组是值类型,每次传递都将产生一份副本。显然这种数据结构无法完全满足开发者的真实需求。Go语言提供了数组切片(slice)来弥补数组的不足。
为了解决这两个缺点,所以才有了切片

切片并不是数组或数组指针,它只是一个数据结构,它通过内部指针和相关属性引用数组片段,以实现变⻓方案。 slice 并不是真正意义上的动态数组,而是一个引用类型。slice 总是指向一个底层 array,slice
的声明也可以像 array 一样,只是不需要长度。

切片的创建和初始化

切片与数组的区别

slice和数组的区别:声明数组时,方括号内写明了数组的长度,而声明slice时,方括号内没有任何字符或者…

// 数组的定义,数组里面的长度是固定的一个常量,数组不能修改长度,len和cap永远都是7
a := [7]int{1, 2, 3, 4, 5, 6, 7}
fmt.Println("slice=", a,"\nlen=", len(a), "\ncap=", cap(a))
// 切片的定义:[]里面为空,或者为...切片的长度或容量可以不固定
b := [] int {}
fmt.Println("b=", b,"\nb=", len(b), "\ncap=", cap(b))// [],0,0
// 给切片末尾追加一个成员,再次打印发现长度和容量均发生了变化
b = append(b, 11)// [11],1,1
// 创建切片并初始化
c := []int{1, 2, 3} 
// 总之一句话,声明slice时,方括号内没有任何数字,或者为...

切片的创建

1、自动推导类型

	// 自动推导类型
	s1 := []int{1, 2, 3, 4}
	fmt.Println("s1= ", s1)

控制台

s1=  [1 2 3 4]

2、借助make函数

借助make函数,格式 make(切片类型,长度,容量)

	s2 := make([]int, 5, 10)
	fmt.Printf("len = %d, cap = %d\n", len(s2), cap(s2))

控制台

len = 5, cap = 10

如果没有指定容量,那么容量和长度一样

	s3 := make([]int, 5)
	fmt.Printf("len = %d, cap = %d\n", len(s3), cap(s3))

控制台

len = 5, cap = 5

切片属性(起始下标、长度、容量)

下面我们来从数组里截取一个切片

array := []int{10, 20, 30, 0, 0}
// array[low,high,max]
slice := array[0:3:5]
fmt.Println("slice=", slice,"\nlen=", len(slice), "\ncap=", cap(slice))

// 控制台打印
slice= [10 20 30] 
len= 3 
cap= 5

上述代码:0,3代表:从数组中的第一个元素开始截取,到索引为3的元素为止(不包括索引为3的元素,即10、20、30),然后切片当前的元素共有10、20、30这三个元素,所以该切片的长度为3。最后一个5表示该切片的容量为5,也就是当前切片在不扩容的情况下最多能放的元素的个数为5个元素,当然了如果再多,切片会自动扩容,新容量为之前容量的2倍。
所以新切片的信息为

// 控制台打印
slice= [10 20 30] 
len= 3 
cap= 5

我们来详细的看看这个array[0:3:5]可以把它看成array[low:high:max]
low:下标起点(切片中第一个元素在原数组中的索引下标)
high:下标的终点(不包括此下标),[a[low], a[high])左闭右开
len = high - low,(长度:切片当前所存放的元素的个数)
cap = max - low, (容量: 切片在不扩容的情况下最多能放的元素的个数)

cap:绝对不会错的cap计算方式,其实切片的容量就是在原数组中从切片的第一个元素开始,到原数组最后一个元素的元素的个数(包括切片的第一个和原数组最后一个元素)。
如下代码,cap就是5 -1 = 4。即切片第一个元素为20,20在原素组中开始数到最后一个元素0为止,即20、30、0、0,一共有4个元素,所以cap为4。此时切片的长度为4-1=3,即当前切片中所存有的元素个数,20、30、0,一共是3个。

array := []int{10, 20, 30, 0, 0}

slice := array[1:4:5]

可以把切片看成一个杯子,长度就是杯子里当前的水量,容量就是杯子体积在不发生变化时最多能盛的水量(这里不太恰当,切片会自动扩容)。

切片的结构有三部分:

  • 1、地址指针
  • 2、长度(切片当前所存放的元素的个数)
  • 3、容量(切片里最多能放的元素的个数)

在这里插入图片描述

切片的操作

切片截取

在这里插入图片描述
示例说明

 array := []int{0, 1, 2, 3, 4, 5, 6, 7, 8, 9}
 // [low:high:max] 取下标从low开始的元素,len=high-low, cap=max-low
 s1 := array[:] // [0:len(array):len(array)] 不指定容量,容量和长度一样

在这里插入图片描述
注:切片的容量:切片的容量就是在原数组中从切片的第一个元素开始,到原数组最后一个元素的元素的个数(包括切片的第一个和原数组最后一个元素)

切片和底层数组关系

修改切片中某个元素的值,对应的原数组中的值也会被改变,所以切片依旧指向原底层数组。
如下代码,注释很详细,这里不再解释

a := []int{1, 2, 3, 4, 5, 6, 7, 8, 9}
	// 新切片
	s1 := a[2:5]
	// s1: [3 4 5] len: 3 cap: 7
	fmt.Println("s1:", s1, "len:", len(s1), "cap:", cap(s1))

	// 修改切片的第一个元素的值
	s1[0] = 666
	// 打印s1: [666 4 5] len: 3 cap: 7
	fmt.Println("s1:", s1, "len:", len(s1), "cap:", cap(s1))
	// 原数组对应的元素也发生了改变
	// a: [1 2 666 4 5 6 7 8 9]
	fmt.Println("a:", a)

	// 创建一个新的切片,该切片是从s1:[3,4,5]的下标索引为2的元素对应原数组的元素5开始至下标为6结束,即5678
	// 故,新切片长度为4,容量为5
	s2:= s1[2:7]
	// 打印:s2: [5 6 7 8 9] len: 5 cap: 5
	fmt.Println("s2:", s2, "len:", len(s2), "cap:", cap(s2))

	// 修改切片s2的元素
	s2[0] = 777
	// 打印 s2: [777 6 7 8 9]
	fmt.Println("s2:", s2)
	// 打印 a: [1 2 666 4 777 6 7 8 9]
	fmt.Println("a:", a)

内建函数

1、append

append函数向 slice 尾部添加数据,返回新的 slice 对象,而且如果切片的容量不够,它还会自动帮切片扩容。

添加数据

	var s1 []int //创建nil切换
    //s1 := make([]int, 0)
	// 在原切片的末尾添加元素
    s1 = append(s1, 1)       //追加1个元素
    s1 = append(s1, 2, 3)    //追加2个元素
    s1 = append(s1, 4, 5, 6) //追加3个元素
    fmt.Println(s1)          //[1 2 3 4 5 6]

    s2 := make([]int, 5)
    s2 = append(s2, 6)
    fmt.Println(s2) //[0 0 0 0 0 6]

    s3 := []int{1, 2, 3}
    s3 = append(s3, 4, 5)
    fmt.Println(s3)//[1 2 3 4 5]

自动扩容

append函数会智能地底层数组的容量增长,一旦超过原底层数组容量,通常以2倍容量重新分配底层数组,并复制原来的数据:
由一下代码可以看出,当添加的元素超过了切片的容量时,append会将切片的容量变为原先的2倍。

s := make([]int, 0, 1)
	for i := 0; i < 50; i++ {
		s = append(s, i)
		fmt.Println("切片中存放的元素个数:", len(s), ", cap:", cap(s))
	}
	/*
	切片中存放的元素个数: 1 , cap: 1
	切片中存放的元素个数: 2 , cap: 2
	切片中存放的元素个数: 3 , cap: 4
	切片中存放的元素个数: 4 , cap: 4
	切片中存放的元素个数: 5 , cap: 8
	切片中存放的元素个数: 6 , cap: 8
	切片中存放的元素个数: 7 , cap: 8
	切片中存放的元素个数: 8 , cap: 8
	切片中存放的元素个数: 9 , cap: 16
	切片中存放的元素个数: 10 , cap: 16
	切片中存放的元素个数: 11 , cap: 16
	切片中存放的元素个数: 12 , cap: 16
	切片中存放的元素个数: 13 , cap: 16
	切片中存放的元素个数: 14 , cap: 16
	切片中存放的元素个数: 15 , cap: 16
	切片中存放的元素个数: 16 , cap: 16
	切片中存放的元素个数: 17 , cap: 32
	切片中存放的元素个数: 18 , cap: 32
	切片中存放的元素个数: 19 , cap: 32
	切片中存放的元素个数: 20 , cap: 32
	切片中存放的元素个数: 21 , cap: 32
	切片中存放的元素个数: 22 , cap: 32
	切片中存放的元素个数: 23 , cap: 32
	切片中存放的元素个数: 24 , cap: 32
	切片中存放的元素个数: 25 , cap: 32
	切片中存放的元素个数: 26 , cap: 32
	切片中存放的元素个数: 27 , cap: 32
	切片中存放的元素个数: 28 , cap: 32
	切片中存放的元素个数: 29 , cap: 32
	切片中存放的元素个数: 30 , cap: 32
	切片中存放的元素个数: 31 , cap: 32
	切片中存放的元素个数: 32 , cap: 32
	切片中存放的元素个数: 33 , cap: 64
	切片中存放的元素个数: 34 , cap: 64
	切片中存放的元素个数: 35 , cap: 64
	切片中存放的元素个数: 36 , cap: 64
	切片中存放的元素个数: 37 , cap: 64
	切片中存放的元素个数: 38 , cap: 64
	切片中存放的元素个数: 39 , cap: 64
	切片中存放的元素个数: 40 , cap: 64
	切片中存放的元素个数: 41 , cap: 64
	切片中存放的元素个数: 42 , cap: 64
	切片中存放的元素个数: 43 , cap: 64
	切片中存放的元素个数: 44 , cap: 64
	切片中存放的元素个数: 45 , cap: 64
	切片中存放的元素个数: 46 , cap: 64
	切片中存放的元素个数: 47 , cap: 64
	切片中存放的元素个数: 48 , cap: 64
	切片中存放的元素个数: 49 , cap: 64
	切片中存放的元素个数: 50 , cap: 64
	*/

2、copy

函数 copy 在两个 slice 间复制数据,复制⻓度以 len 小的为准,两个 slice 可指向同⼀底层数组。
就是将短的切片的元素替换掉长的切片的部分元素
如下代码

	srcSlice := []int{1, 2}
	dstSlice := []int{6, 6, 6, 6, 6}
	// 将srcSlice复制到dstSlice里
	copy(dstSlice, srcSlice)
	// dst=  [1 2 6 6 6]
	fmt.Println("dst= ", dstSlice)
	// dst=  [1 2]
	fmt.Println("dst= ", srcSlice)
切片做函数参数

数组在做函数参数的时候是值传递,而切片做函数参数时是引用传递

个人学习笔记,不喜勿喷