简介:
append()用来将元素添加到切片末尾并返回结果。
调用append函数必须用原来的切片变量接收返回值
append追加元素,如果slice还有容量的话,就会将新的元素放在原来slice后面的剩余空间里,当底层数组装不下的时候,Go就会创建新的底层数组来保存这个切片,slice地址也随之改变。
分配了新的地址后,再把原来slice中的元素逐个拷贝到新的slice中,并返回。
(1) append()追加一个元素
slice = append(slice,elem1,elem2)
append括号内,第一个参数slice后可以加多个参数。
package main
import "fmt"
//切片进阶操作
func main(){
//append()为切片追加元素
s1 := []string {"火鸡面","辛拉面","汤达人"}
fmt.Printf("s1=%v len(s1)=%d cap(s1)=%d\n",s1,len(s1),cap(s1))
//调用append函数必须用原来的切片变量接收返回值
s1 = append(s1,"小当家") //append追加元素,原来的底层数组装不下的时候,Go就会创建新的底层数组来保存这个切片
fmt.Printf("s1=%v len(s1)=%d cap(s1)=%d\n",s1,len(s1),cap(s1))//cap增加两倍
}
输出结果:
s1=[火鸡面 辛拉面 汤达人] len(s1)=3 cap(s1)=3
s1=[火鸡面 辛拉面 汤达人 小当家] len(s1)=4 cap(s1)=6
(2)append()追加一个切片
slice = append(slice,anotherSlice…)
append括号内只能有两个参数,一个切片,另一个追加的切片。
package main
import "fmt"
//切片进阶操作
func main(){
//append()为切片追加元素
s1 := []string {"火鸡面","辛拉面","汤达人"}
fmt.Printf("s1=%v len(s1)=%d cap(s1)=%d\n",s1,len(s1),cap(s1))
//调用append函数必须用原来的切片变量接收返回值
s1 = append(s1,"小当家") //append动态追加元素,原来的底层数组容纳不下足够多的元素时,切片就会开始扩容,Go底层数组就会把底层数组换一个
fmt.Printf("s1=%v len(s1)=%d cap(s1)=%d\n",s1,len(s1),cap(s1))
//调用append添加一个切片
s2 := []string{"脆司令","圣斗士"}
s1 = append(s1,s2...)//...表示拆开切片,再添加
fmt.Printf("s1=%v len(s1)=%d cap(s1)=%d",s1,len(s1),cap(s1))
}
输出结果:
s1=[火鸡面 辛拉面 汤达人] len(s1)=3 cap(s1)=3
s1=[火鸡面 辛拉面 汤达人 小当家] len(s1)=4 cap(s1)=6
s1=[火鸡面 辛拉面 汤达人 小当家 脆司令 圣斗士] len(s1)=6 cap(s1)=6
(3)使用make创建切片时,用append()添加元素常犯错误
package main
import "fmt"
func main(){
var a = make([]int, 5, 10)
fmt.Println(a)
fmt.Printf("%p\n",a)
for i := 0; i <10; i++ {
a = append(a,i)
//%p 打印切片地址
fmt.Printf("%v,%p,cap(a):%d\n",a,a,cap(a))
}
}
输出结果:
[0 0 0 0 0]
0xc0000180a0
[0 0 0 0 0 0],0xc0000180a0,cap(a):10
[0 0 0 0 0 0 1],0xc0000180a0,cap(a):10
[0 0 0 0 0 0 1 2],0xc0000180a0,cap(a):10
[0 0 0 0 0 0 1 2 3],0xc0000180a0,cap(a):10
[0 0 0 0 0 0 1 2 3 4],0xc0000180a0,cap(a):10
[0 0 0 0 0 0 1 2 3 4 5],0xc00007c000,cap(a):20
[0 0 0 0 0 0 1 2 3 4 5 6],0xc00007c000,cap(a):20
[0 0 0 0 0 0 1 2 3 4 5 6 7],0xc00007c000,cap(a):20
[0 0 0 0 0 0 1 2 3 4 5 6 7 8],0xc00007c000,cap(a):20
[0 0 0 0 0 0 1 2 3 4 5 6 7 8 9],0xc00007c000,cap(a):20
注意:
(1)make创建切片,有默认长度,就有默认值。
append()再添加元素,是在默认值后添加,而不是覆盖默认值。
(2)当元素超过make创建时设的容量10时,原底层数组装不下,就会换一段新的连续的地址来存放元素。
(3)利用append删除元素
Go中没有提供专门删除元素的函数,而是通过切片本身的特点来删除元素。
即以被删除元素为分界点,再利用append将前后两个部分的内存重新连接起来。
例如:
如果要在切片s中删除一个元素,被删除的元素索引为index
则删除过程为
s = append ( s[ :index ], s[ index+1: ] )
将前后两个部分重新连接起来,实质上就是将删除点的元素前移,将内存重新连接起来。
package main
import "fmt"
func main(){
a1 := [...]int{1,2,5,3,78,9,4,9,23,32}
s1 := a1[:] //得到切片
fmt.Println(s1)
//删除索引为4的78
s1 = append(s1[:4],s1[5:]...)
fmt.Println(s1)
fmt.Println(a1)
}
Go中利用append删除元素的原理为:
输出结果:
[1 2 5 3 78 9 4 9 23 32]
[1 2 5 3 9 4 9 23 32]
[1 2 5 3 9 4 9 23 32 32]
理解后可以试着猜猜下面的程序输出是什么:
package main
import "fmt"
func main(){
a1 := [...]int{1,2,5,3,78,9,4,9,23,32}
s1 := a1[:] //得到切片
fmt.Println(s1)
//删掉索引为2和3的5,3
s1 = append(s1[:2],s1[4:]...)
fmt.Println(s1)
fmt.Println(a1)
}
作者:苏州干饭queen
链接:https://juejin.cn/post/6951672096699187207