本文介绍了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]

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。

标签:
Golang,slice切片操作, golang,slice,删除,golang,切片追加

免责声明:本站文章均来自网站采集或用户投稿,网站不提供任何软件下载或自行开发的软件! 如有用户或公司发现本站内容信息存在侵权行为,请邮件告知! 858582#qq.com
桃源资源网 Design By www.nqtax.com

《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线

暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。

艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。

《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。