本文介绍了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:true Success: 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 []string 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); } --- 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]
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持小牛知识库。
如果想增加切片的容量,我们必须创建一个新的更大的切片并把原分片的内容都拷贝过来。下面的代码描述了从拷贝切片的 copy 函数和向切片追加新元素的 append 函数。 示例 7.12 copy_append_slice.go package main import "fmt" func main() { sl_from := []int{1, 2, 3} sl_to := mak
问题内容: 这个带有添加功能的删除技巧如何工作? 似乎它正在抓取第一个元素(空数组)之前的所有内容 然后将所有内容附加在第一个元素之后(位置零) …(点对点)有什么作用? 问题答案: 切片在哪里,是要删除的元素的索引: 是Go中可变参数的语法。 基本上,在 定义 函数时,它将把您传递的所有参数放入该类型的一个切片中。这样,您可以传递任意数量的参数(例如,可以传递任意数量的参数)。 现在,当 调用
问题内容: 我进行了以下测试,即打印原始输入切片(过滤后),而没有删除元素,但是最后有一个额外的元素,使输入切片具有相同的长度,即使过滤后应该较短。 我已经看过这篇文档https://github.com/golang/go/wiki/SliceTricks#delete 但是我认为我缺少有关Go的一些技巧,因为似乎我使用的切片方法错误。 如何避免出现“输出片段”?(以正确的方式打印,包含正确的元
问题内容: 我很好奇拆包切片并将其作为参数发送给可变参数函数。 假设我们有一个带有可变参数的函数: 如果我们不想传入一个接口,它就可以工作,那么我们是否拆包都没关系: 如果我们有一片片的话,那会很棘手。在这里,编译器不允许我们传递解压版本: 错误提示: 在解包参数中不能将sliceOfSlices(类型[] [] interface {})用作类型[] interface {} 我不知道为什么会这
问题内容: 在Golang中,和之间有什么区别? 我发现这两种方法都可行,但是哪一种更好? 问题答案: 除了fabriziom的答案外,您还可以在“转到切片:用法和内部方法”中看到更多示例,其中提到了的用法: 由于slice()的零值就像 零长度的slice一样 ,您可以声明slice变量,然后在循环中附加到该变量: 这意味着,要附加到切片,您不必先分配内存:切片足以作为要添加的切片。
问题内容: 为了使切片追加操作更快,我们需要分配足够的容量。有两种附加切片的方法,下面是代码: 结果是: BenchmarkSliceAppend-4 200000000 7.87 ns / op 8 B / op 0 allocs / op BenchmarkSliceSet-4 300000000 5.76 ns / op 8 B / op 比我快,我想知道为什么吗? 问题答案: 只需将值分配