本文介紹了golang/118375.html">golang/286467.html">Golang slice切片操作之切片的追加、刪除、插入等,分享給大家,具體如下:
一、一般操作
1,聲明變量,go自動初始化為nil,長度:0,地址:0,nil
func main(){ var ss []string; fmt.Printf("length:%v /taddr:%p /tisnil:%v",len(ss),ss, ss==nil) }---Running...length:0 addr:0x0 isnil:trueSuccess: process exited with code 0.
2,切片的追加,刪除,插入操作
func main(){ var ss []string; fmt.Printf("[ local print ]/t:/t length:%v/taddr:%p/tisnil:%v/n",len(ss),ss, ss==nil) print("func print",ss) //切片尾部追加元素append elemnt for i:=0;i<10;i++{ ss=append(ss,fmt.Sprintf("s%d",i)); } fmt.Printf("[ local print ]/t:/tlength:%v/taddr:%p/tisnil:%v/n",len(ss),ss, ss==nil) print("after append",ss) //刪除切片元素remove element at index index:=5; ss=append(ss[:index],ss[index+1:]...) print("after delete",ss) //在切片中間插入元素insert element at index; //注意:保存后部剩余元素,必須新建一個臨時切片 rear:=append([]string{},ss[index:]...) ss=append(ss[0:index],"inserted") ss=append(ss,rear...) print("after insert",ss)}func print(msg string,ss []string){ fmt.Printf("[ %20s ]/t:/tlength:%v/taddr:%p/tisnil:%v/tcontent:%v",msg,len(ss),ss, ss==nil,ss) fmt.Println()}------Running...[ local print ] : length:0 addr:0x0 isnil:true[ func print ] : length:0 addr:0x0 isnil:true content:[][ local print ] : length:10 addr:0xc208056000 isnil:false[ after append ] : length:10 addr:0xc208056000 isnil:false content:[s0 s1 s2 s3 s4 s5 s6 s7 s8 s9][ after delete ] : length:9 addr:0xc208056000 isnil:false content:[s0 s1 s2 s3 s4 s6 s7 s8 s9][ after insert ] : length:10 addr:0xc208056000 isnil:false content:[s0 s1 s2 s3 s4 inserted s6 s7 s8 s9]Success: process exited with code 0.
3,copy的使用。
在使用copy復制切片之前,要保證目標切片有足夠的大小,注意是大小,而不是容量,還是看例子:
func main() { var sa = make ([]string,0); for i:=0;i<10;i++{ sa=append(sa,fmt.Sprintf("%v",i)) } var da =make([]string,0,10); var cc=0; cc= copy(da,sa); fmt.Printf("copy to da(len=%d)/t%v/n",len(da),da) da = make([]string,5) cc=copy(da,sa); fmt.Printf("copy to da(len=%d)/tcopied=%d/t%v/n",len(da),cc,da) da = make([]string,10) cc =copy(da,sa); fmt.Printf("copy to da(len=%d)/tcopied=%d/t%v/n",len(da),cc,da) }---Running...copy to da(len=0) []copy to da(len=5) copied=5 [0 1 2 3 4]copy to da(len=10) copied=10 [0 1 2 3 4 5 6 7 8 9]
從上面運行結果,明顯看出,目標切片大小0,容量10,copy不能復制。目標切片大小小于源切片大小,copy就按照目標切片大小復制,不會報錯。
二、初始大小和容量
當我們使用make初始化切片的時候,必須給出size。go語言的書上一般都會告訴我們,當切片有足夠大小的時候,append操作是非??斓?。但是當給出初始大小后,我們得到的實際上是一個含有這個size數量切片類型的空元素,看例子:
func main(){ var ss=make([]string,10); ss=append(ss,"last"); print("after append",ss) }---Running...[ after append ] : length:11 addr:0xc20804c000 isnil:false content:[ last]
實際上,此時我們應該先用下標為切片元素負值。但是如果我們既想有好的效率,有想繼續使用append函數而不想區分是否有空的元素,此時就要請出make的第三個參數,容量,也就是我們通過傳遞給make,0的大小和足夠大的容量數值就行了。
func main(){ var ss=make([]string,0,10); ss=append(ss,"last"); print("after append",ss) }---Running...[ after append ] : length:1 addr:0xc20804a000 isnil:false content:[last]
三、切片的指針。
1,當我們用append追加元素到切片時,如果容量不夠,go就會創建一個新的切片變量,看下面程序的執行結果:
func main() { var sa []stringfmt.Printf("addr:%p /t/tlen:%v content:%v/n",sa,len(sa),sa); for i:=0;i<10;i++{ sa=append(sa,fmt.Sprintf("%v",i)) fmt.Printf("addr:%p /t/tlen:%v content:%v/n",sa,len(sa),sa); } fmt.Printf("addr:%p /t/tlen:%v content:%v/n",sa,len(sa),sa);}---Running ...addr:0x0 len:0 content:[]addr:0x1030e0c8 len:1 content:[0]addr:0x10328120 len:2 content:[0 1]addr:0x10322180 len:3 content:[0 1 2]addr:0x10322180 len:4 content:[0 1 2 3]addr:0x10342080 len:5 content:[0 1 2 3 4]addr:0x10342080 len:6 content:[0 1 2 3 4 5]addr:0x10342080 len:7 content:[0 1 2 3 4 5 6]addr:0x10342080 len:8 content:[0 1 2 3 4 5 6 7]addr:0x10324a00 len:9 content:[0 1 2 3 4 5 6 7 8]addr:0x10324a00 len:10 content:[0 1 2 3 4 5 6 7 8 9]addr:0x10324a00 len:10 content:[0 1 2 3 4 5 6 7 8 9]//很明顯,切片的地址經過了數次改變。
2,如果,在make初始化切片的時候給出了足夠的容量,append操作不會創建新的切片:
func main() { var sa = make ([]string,0,10);fmt.Printf("addr:%p /t/tlen:%v content:%v/n",sa,len(sa),sa); for i:=0;i<10;i++{ sa=append(sa,fmt.Sprintf("%v",i)) fmt.Printf("addr:%p /t/tlen:%v content:%v/n",sa,len(sa),sa); } fmt.Printf("addr:%p /t/tlen:%v content:%v/n",sa,len(sa),sa);}
addr:0x10304140 len:0 content:[]addr:0x10304140 len:1 content:[0]addr:0x10304140 len:2 content:[0 1]addr:0x10304140 len:3 content:[0 1 2]addr:0x10304140 len:4 content:[0 1 2 3]addr:0x10304140 len:5 content:[0 1 2 3 4]addr:0x10304140 len:6 content:[0 1 2 3 4 5]addr:0x10304140 len:7 content:[0 1 2 3 4 5 6]addr:0x10304140 len:8 content:[0 1 2 3 4 5 6 7]addr:0x10304140 len:9 content:[0 1 2 3 4 5 6 7 8]addr:0x10304140 len:10 content:[0 1 2 3 4 5 6 7 8 9]addr:0x10304140 len:10 content:[0 1 2 3 4 5 6 7 8 9]//可見,切片的地址一直保持不變
3, 如果不能準確預估切片的大小,又不想改變變量(如:為了共享數據的改變),這時候就要請出指針來幫忙了,下面程序中,sa就是osa這個切片的指針,我們共享切片數據和操作切片的時候都使用這個切片地址就ok了,其本質上是:append操作亦然會在需要的時候構造新的切片,不過是將地址都保存到了sa中,因此我們通過該指針始終可以訪問到真正的數據。
func main() { var osa = make ([]string,0); sa:=&osa; for i:=0;i<10;i++{ *sa=append(*sa,fmt.Sprintf("%v",i)) fmt.Printf("addr of osa:%p,/taddr:%p /t content:%v/n",osa,sa,sa); } fmt.Printf("addr of osa:%p,/taddr:%p /t content:%v/n",osa,sa,sa); }---Running...addr of osa:0xc20800a220, addr:0xc20801e020 content:&[0]addr of osa:0xc20801e0a0, addr:0xc20801e020 content:&[0 1]addr of osa:0xc20803e0c0, addr:0xc20801e020 content:&[0 1 2]addr of osa:0xc20803e0c0, addr:0xc20801e020 content:&[0 1 2 3]addr of osa:0xc208050080, addr:0xc20801e020 content:&[0 1 2 3 4]addr of osa:0xc208050080, addr:0xc20801e020 content:&[0 1 2 3 4 5]addr of osa:0xc208050080, addr:0xc20801e020 content:&[0 1 2 3 4 5 6]addr of osa:0xc208050080, addr:0xc20801e020 content:&[0 1 2 3 4 5 6 7]addr of osa:0xc208052000, addr:0xc20801e020 content:&[0 1 2 3 4 5 6 7 8]addr of osa:0xc208052000, addr:0xc20801e020 content:&[0 1 2 3 4 5 6 7 8 9]addr of osa:0xc208052000, addr:0xc20801e020 content:&[0 1 2 3 4 5 6 7 8 9]
以上就是本文的全部內容,希望對大家的學習有所幫助,也希望大家多多支持VEVB武林網。
新聞熱點
疑難解答