Java转Go—07切片

切片

在go语言中,切片是对数组的抽象,数组在声明时指定了长度之后就不可再进行改变,在特定场景下数组就不适用,所以就有了切片类型,切片就是“动态数组”,和数组相比,切片的长度是不固定的,可以在切片后面追加元素,长度自动扩容。

切片定义

定义切片和定义数组很像,区别就是定义一个切片不需要指定长度。

var 切片名 []类型

切片还可以使用make函数定义。

make([]T, length, capacity)

make函数有三个参数:

  1. 第一个参数为切片类型,可以是[]int,[]string,[]float32等。
  2. 第二个参数为切片初始长度。
  3. 第三个为切片容量,该参数为可选参数。

切片初始化

一个切片在初始化之前为空切片(nil),长度为0,可以在声明切片时直接初始化切片,如下表示声明一个int切片,初始化值为{1, 2, 3}

s :=[] int {1,2,3 }

初始化为数组的引用,假设有一个数组arr,在初始化为数组的引用时,通过开始索引和结束索引控制初始化的切片大小和切片内元素个数。

s := arr[startIndex:endIndex] // 从 startIndex 到 endIndex - 1 初始化为一个切片
s := arr[startIndex:]   // 从 startIndex 到 数组结尾 初始化为一个切片
s := arr[:endIndex]    // 从 数组开始 到 endIndex - 1 初始化为一个切片
s := arr[:]      // 从 数组开始 到 数组结尾 初始化为一个切片(整个数组)

代码示例:

package main

import "fmt"

func main() {
 arr := [5]int{12345}
 s1 := arr[:]
 s2 := arr[2:]
 s3 := arr[:3]
 s4 := arr[1:4]
 fmt.Println(s1)
 fmt.Println(s2)
 fmt.Println(s3)
 fmt.Println(s4)
}

运行结果:

Java转Go—07切片
运行结果

append和copy

append表示在一个切片的末尾追加元素。

copy表示复制一个切片里面的元素到另一个切片。

代码示例:

package main

import "fmt"

func main() {
 s := []int{1,2,3}
 fmt.Println("切片s:",s)
 s = append(s, 4)     // 添加一个元素4到切片s中
 fmt.Println("切片s:",s)
 s1 := make([]int4)
 copy(s1, s)           // 将切片s的内容拷贝到切片s1中
 fmt.Println("切片s1:",s1)
}

运行结果:

Java转Go—07切片
运行结果

切片截取

切片截取使用中括号[],通过指定需要截取的开始索引和结束索引。

代码示例:

package main

import "fmt"

func main() {
 s := []int{1,2,3,4,5}
 fmt.Println("完整切片:", s)
 fmt.Println("s[1:3]:", s[1:3])   // 截取索引1(包含)到索引3(不包含)
 fmt.Println("s[:4]:", s[:4])   // 默认开始索引为0
 fmt.Println("s[2:]:", s[2:])   // 默认结束索引为len
}

运行结果:

Java转Go—07切片
运行结果

len和cap

长度和容量区别:

  • 长度:长度表示切片中实际存储的元素个数
  • 容量:容量表示切片底层使用的数组的大小

当定义一个切片时,如果没有通过make方法指定cap,则底层会申请一个和切片长度一样的数组,这个数组的大小就是cap,当使用append朝切片中追加元素时,如果追加元素后新的len小于cap,则底层数组不会改变,当新的len大于cap时,底层就会重新申请一个数组,且数组的长度为cap * 2,然后将之前数组的元素全部复制到新数组中。

代码示例:

package main

import "fmt"

func main() {
 s := make([]int3)
 fmt.Printf("len:%d, cap:%d n"len(s), cap(s))  // 此时len=3,cap=3

 s = append(s, 1)
 fmt.Printf("len:%d, cap:%d n"len(s), cap(s))  // append一个元素,len=4,大于cap,所以底层数组扩容,cap=6

 s = append(s, 1)
 fmt.Printf("len:%d, cap:%d n"len(s), cap(s))  // append一个元素,len=5,小于cap,底层数组不变,cap=6

 s = append(s, 1)
 fmt.Printf("len:%d, cap:%d n"len(s), cap(s))  // append一个元素,len=6,小于cap,底层数组不变,cap=6

 s = append(s, 1)
 fmt.Printf("len:%d, cap:%d n"len(s), cap(s))  // append一个元素,len=7,大于cap,所以底层数组扩容,cap=12
}

运行结果:

Java转Go—07切片
运行结果


原文始发于微信公众号(良猿):Java转Go—07切片

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 举报,一经查实,本站将立刻删除。

文章由极客之音整理,本文链接:https://www.bmabk.com/index.php/post/215862.html

(0)
小半的头像小半

相关推荐

发表回复

登录后才能评论
极客之音——专业性很强的中文编程技术网站,欢迎收藏到浏览器,订阅我们!