在go語言中,append()函數(shù)用于為切片動態(tài)添加元素,可以將元素添加到切片末尾并返回結(jié)果;調(diào)用append函數(shù)必須用原來的切片變量接收返回值,追加一個元素可使用“slice = append(slice,elem1,elem2)”語句,追加一個切片可使用“slice = append(slice,anotherSlice…)”語句。
本教程操作環(huán)境:windows7系統(tǒng)、GO 1.18版本、Dell G3電腦。
Go語言的內(nèi)建函數(shù) append() 可以為切片動態(tài)添加元素。
要點:
- append()用來將元素添加到切片末尾并返回結(jié)果。
- 調(diào)用append函數(shù)必須用原來的切片變量接收返回值
- append追加元素,如果slice還有容量的話,就會將新的元素放在原來slice后面的剩余空間里,當(dāng)?shù)讓訑?shù)組裝不下的時候,Go就會創(chuàng)建新的底層數(shù)組來保存這個切片,slice地址也隨之改變。
- 分配了新的地址后,再把原來slice中的元素逐個拷貝到新的slice中,并返回。
(1) append()追加一個元素
slice = append(slice,elem1,elem2)
append括號內(nèi),第一個參數(shù)slice后可以加多個參數(shù)。
package main import "fmt" //切片進(jìn)階操作 func main(){ //append()為切片追加元素 s1 := []string {"火雞面","辛拉面","湯達(dá)人"} fmt.Printf("s1=%v len(s1)=%d cap(s1)=%dn",s1,len(s1),cap(s1)) //調(diào)用append函數(shù)必須用原來的切片變量接收返回值 s1 = append(s1,"小當(dāng)家") //append追加元素,原來的底層數(shù)組裝不下的時候,Go就會創(chuàng)建新的底層數(shù)組來保存這個切片 fmt.Printf("s1=%v len(s1)=%d cap(s1)=%dn",s1,len(s1),cap(s1))//cap增加兩倍 }
輸出結(jié)果:
s1=[火雞面 辛拉面 湯達(dá)人] len(s1)=3 cap(s1)=3 s1=[火雞面 辛拉面 湯達(dá)人 小當(dāng)家] len(s1)=4 cap(s1)=6
(2)append()追加一個切片
slice = append(slice,anotherSlice...)
append括號內(nèi)只能有兩個參數(shù),一個切片,另一個追加的切片。
package main import "fmt" //切片進(jìn)階操作 func main(){ //append()為切片追加元素 s1 := []string {"火雞面","辛拉面","湯達(dá)人"} fmt.Printf("s1=%v len(s1)=%d cap(s1)=%dn",s1,len(s1),cap(s1)) //調(diào)用append函數(shù)必須用原來的切片變量接收返回值 s1 = append(s1,"小當(dāng)家") //append動態(tài)追加元素,原來的底層數(shù)組容納不下足夠多的元素時,切片就會開始擴(kuò)容,Go底層數(shù)組就會把底層數(shù)組換一個 fmt.Printf("s1=%v len(s1)=%d cap(s1)=%dn",s1,len(s1),cap(s1)) //調(diào)用append添加一個切片 s2 := []string{"脆司令","圣斗士"} s1 = append(s1,s2...)//...表示拆開切片,再添加 fmt.Printf("s1=%v len(s1)=%d cap(s1)=%d",s1,len(s1),cap(s1)) }
輸出結(jié)果:
s1=[火雞面 辛拉面 湯達(dá)人] len(s1)=3 cap(s1)=3 s1=[火雞面 辛拉面 湯達(dá)人 小當(dāng)家] len(s1)=4 cap(s1)=6 s1=[火雞面 辛拉面 湯達(dá)人 小當(dāng)家 脆司令 圣斗士] len(s1)=6 cap(s1)=6
(3)使用make創(chuàng)建切片時,用append()添加元素常犯錯誤
package main import "fmt" func main(){ var a = make([]int, 5, 10) fmt.Println(a) fmt.Printf("%pn",a) for i := 0; i <10; i++ { a = append(a,i) //%p 打印切片地址 fmt.Printf("%v,%p,cap(a):%dn",a,a,cap(a)) } }
輸出結(jié)果:
[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創(chuàng)建切片,有默認(rèn)長度,就有默認(rèn)值。
append()再添加元素,是在默認(rèn)值后添加,而不是覆蓋默認(rèn)值。
(2)當(dāng)元素超過make創(chuàng)建時設(shè)的容量10時,原底層數(shù)組裝不下,就會換一段新的連續(xù)的地址來存放元素。
(4)利用append刪除元素
Go中沒有提供專門刪除元素的函數(shù),而是通過切片本身的特點來刪除元素。
即以被刪除元素為分界點,再利用append將前后兩個部分的內(nèi)存重新連接起來。
例如:
如果要在切片s中刪除一個元素,被刪除的元素索引為index
則刪除過程為
s = append ( s[ :index ], s[ index+1: ] )
將前后兩個部分重新連接起來,實質(zhì)上就是將刪除點的元素前移,將內(nèi)存重新連接起來。
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刪除元素的原理為:
(畫的有毫毫丑…..)
輸出結(jié)果:
[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) }
正確結(jié)果:
[1 2 5 3 78 9 4 9 23 32] [1 2 78 9 4 9 23 32] [1 2 78 9 4 9 23 32 23 32]
【