本文主要介绍Go语言中切片(slice)及它的基本使用。

引子

因为数组的长度是固定的并且数组长度属于类型的一部分,所以数组有很多的局限性。 例如:

  1. func arraySum(x []int) int{
  2. sum :=
  3. for _, v := range x{
  4. sum = sum + v
  5. }
  6. return sum
  7. }

这个求和函数只能接受[3]int类型,其他的都不支持。 再比如,

  1. a := []int{, , }

数组a中已经有三个元素了, 我们不能再及所需往数组a中添加新元素了。

切片

切片(Slice)是一个拥有相同类型元素的可变长度的序列。它是基于数组类型做的一层封装。它非常灵活,支持自动扩容。

切片是一个引用类型,它的内部结构包含地址长度容量。切片一般用于快速地操作一块数据集合。

切片的定义

声明切片类型的基本语法如下:

  1. var name []T

其中:

  name: 表示变量名

  T: 表示切片中的元素类型

举个例子:

  1. func main() {
  2. // 声明切片类型
  3. var a []string //声明一个字符串切片
  4. var b = []int{} //声明一个整型切片并初始化
  5. var c = []bool{false, true} //声明一个布尔切片并初始化
  6. var d = []bool{false, true} //声明一个布尔切片并初始化
  7. fmt.Println(a) //[]
  8. fmt.Println(b) //[]
  9. fmt.Println(c) //[false true]
  10. fmt.Println(a == nil) //true
  11. fmt.Println(b == nil) //false
  12. fmt.Println(c == nil) //false
  13. // fmt.Println(c == d) //切片是引用类型,不支持直接比较,只能和nil比较
  14. }

切片的长度和容量

切片拥有自己的长度和容量,我们可以通过使用内置的len()函数求长度,使用内置的cap()函数求切片的容量。

基于数组定义切片

由于切片的底层就是一个数组,所以我们可以基于数组定义切片。

  1. func main() {
  2. // 基于数组定义切片
  3. a := []int{, , , , }
  4. b := a[:] //基于数组a创建切片,包括元素a[1],a[2],a[3] 包头不包尾
  5. fmt.Println(b) //[56 57 58]
  6. fmt.Printf("type of b:%T\n", b) //type of b:[]int
  7. }

还支持如下方式:

  1. c := a[:] //[56 57 58 59]
  2. d := a[:] //[55 56 57]
  3. e := a[:] //[55 56 57 58 59]

切片再切片

除了基于数组得到切片,我们还可以通过切片来得到切片。

  1. func main() {
  2. //切片再切片
  3. a := [...]string{"北京", "上海", "广州", "深圳", "成都", "重庆"}
  4. fmt.Printf("a:%v type:%T len:%d cap:%d\n", a, a, len(a), cap(a))
  5. b := a[:]
  6. fmt.Printf("b:%v type:%T len:%d cap:%d\n", b, b, len(b), cap(b))
  7. c := b[:]
  8. fmt.Printf("c:%v type:%T len:%d cap:%d\n", c, c, len(c), cap(c))
  9. }

输出:

  1. a:[北京 上海 广州 深圳 成都 重庆] type:[]string len: cap:
  2. b:[上海 广州] type:[]string len: cap:
  3. c:[广州 深圳 成都 重庆] type:[]string len: cap:

注意: 对切片进行再切片时,索引不能超过原数组的长度,否则会出现索引越界的错误。

使用make()函数构造切片

我们上面都是基于数组来创建的切片,如果需要动态的创建一个切片,我们就需要使用内置的make()函数,格式如下:

  1. make([]T, size, cap)

其中:

  T: 切片的元素类型

  size:切片中元素的数量

  cap: 切片的容量

举个例子:

  1. func main() {
  2. a := make([]int, , )
  3. fmt.Println(a) //[0 0]
  4. fmt.Println(len(a)) //
  5. fmt.Println(cap(a)) //
  6. }

上面代码中a的内部存储空间已经分配了10个,但实际上只用了2个。 容量并不会影响当前元素的个数,所以len(a)返回2,cap(a)则返回该切片的容量。

切片的本质

切片的本质就是对底层数组的封装,它包含了三个信息:底层数组的指针、切片的长度(len)和切片的容量(cap)。

举个例子,现在有一个数组a := [8]int{0, 1, 2, 3, 4, 5, 6, 7},切片s1 := a[:5],相应示意图如下。

切片s2 := a[3:6],相应示意图如下:

切片不能直接比较

切片之间是不能比较的,我们不能使用==操作符来判断两个切片是否含有全部相等元素。 切片唯一合法的比较操作是和nil比较。 一个nil值的切片并没有底层数组,一个nil值的切片的长度和容量都是0。但是我们不能说一个长度和容量都是0的切片一定是nil,例如下面的示例:

  1. var s1 []int //len(s1)=0;cap(s1)=0;s1==nil
  2. s2 := []int{} //len(s2)=0;cap(s2)=0;s2!=nil
  3. s3 := make([]int, ) //len(s3)=0;cap(s3)=0;s3!=nil

所以要判断一个切片是否是空的,要是用len(s) == 0来判断,不应该使用s == nil来判断。

切片的赋值拷贝

下面的代码中演示了拷贝前后两个变量共享底层数组,对一个切片的修改会影响另一个切片的内容,这点需要特别注意。

  1. func main() {
  2. s1 := make([]int, ) //[0 0 0]
  3. s2 := s1 // 将s1直接赋值给s2,s1和s2 共用一个底层数组
  4. s2[] =
  5. fmt.Println(s1) // [100 0 0]
  6. fmt.Println(s2) // [100 0 0]
  7. }

切片遍历

切片的遍历方式和数组是一致的,支持索引遍历和for range遍历。

  1. func main() {
  2. s := []int{, , }
  3.  
  4. for i := ; i < len(s); i++ {
  5. fmt.Println(i, s[i])
  6. }
  7.  
  8. for index, value := range s {
  9. fmt.Println(index, value)
  10. }
  11. }

append()方法为切片添加元素

Go语言的内建函数append()可以为切片动态添加元素。 每个切片会指向一个底层数组,这个数组能容纳一定数量的元素。当底层数组不能容纳新增的元素时,切片就会自动按照一定的策略进行“扩容”,此时该切片指向的底层数组就会更换。“扩容”操作往往发生在append()函数调用时。 举个例子:

  1. func main() {
  2. //append()添加元素和切片扩容
  3. var numSlice []int
  4. for i := ; i < ; i++ {
  5. numSlice = append(numSlice, i)
  6. fmt.Printf("%v len:%d cap:%d ptr:%p\n", numSlice, len(numSlice), cap(numSlice), numSlice)
  7. }
  8. }

输出:

  1. [] len: cap: ptr:0xc0000a8000
  2. [ ] len: cap: ptr:0xc0000a8040
  3. [ ] len: cap: ptr:0xc0000b2020
  4. [ ] len: cap: ptr:0xc0000b2020
  5. [ ] len: cap: ptr:0xc0000b6000
  6. [ ] len: cap: ptr:0xc0000b6000
  7. [ ] len: cap: ptr:0xc0000b6000
  8. [ ] len: cap: ptr:0xc0000b6000
  9. [ ] len: cap: ptr:0xc0000b8000
  10. [ ] len: cap: ptr:0xc0000b8000

从上面的结果可以看出:

  1. append()函数将元素追加到切片的最后并返回该切片。
  2. 切片numSlice的容量按照1,2,4,8,16这样的规则自动进行扩容,每次扩容后都是扩容前的2倍。

append()函数还支持一次性追加多个元素。 例如:

  1. var citySlice []string
  2. // 追加一个元素
  3. citySlice = append(citySlice, "北京")
  4. // 追加多个元素
  5. citySlice = append(citySlice, "上海", "广州", "深圳")
  6. // 追加切片
  7. a := []string{"成都", "重庆"}
  8. citySlice = append(citySlice, a...)
  9. fmt.Println(citySlice) //[北京 上海 广州 深圳 成都 重庆]

切片的扩容策略

可以通过查看$GOROOT/src/runtime/slice.go源码,其中扩容相关代码如下:

  1. newcap := old.cap
  2. doublecap := newcap + newcap
  3. if cap > doublecap {
  4. newcap = cap
  5. }else {
  6. if old.len < {
  7. newcap = doublecap
  8. }else{
  9. // Check 0 < newcap to detect overflow
  10. // and prevent an infinite loop.
  11. for < newcap && newcap < cap {
  12. newcap += newcap /
  13. }
  14. // Set newcap to the requested cap when
  15. // the newcap calculation overflowed.
  16. if newcap <= {
  17. newcap = cap
  18. }
  19. }
  20. }

从上面的代码可以看出以下内容:

  • 首先判断,如果新申请容量(cap)大于2倍的旧容量(old.cap),最终容量(newcap)就是新申请的容量(cap)。
  • 否则判断,如果旧切片的长度小于1024,则最终容量(newcap)就是旧容量(old.cap)的两倍,即(newcap=doublecap),
  • 否则判断,如果旧切片长度大于等于1024,则最终容量(newcap)从旧容量(old.cap)开始循环增加原来的1/4,即(newcap=old.cap,for {newcap += newcap/4})直到最终容量(newcap)大于等于新申请的容量(cap),即(newcap >= cap)
  • 如果最终容量(cap)计算值溢出,则最终容量(cap)就是新申请容量(cap)。

需要注意的是,切片扩容还会根据切片中元素的类型不同而做不同的处理,比如intstring类型的处理方式就不一样。

使用copy()函数复制切片

首先我们来看一个问题:

  1. func main() {
  2. a := []int{, , , , }
  3. b := a
  4. fmt.Println(a) //[1 2 3 4 5]
  5. fmt.Println(b) //[1 2 3 4 5]
  6. b[] =
  7. fmt.Println(a) //[1000 2 3 4 5]
  8. fmt.Println(b) //[1000 2 3 4 5]
  9. }

由于切片是引用类型,所以a和b其实都指向了同一块内存地址。修改b的同时a的值也会发生变化。

Go语言内建的copy()函数可以迅速地将一个切片的数据复制到另外一个切片空间中,copy()函数的使用格式如下:

  1. copy(destSlice, srcSlice []T)

其中:

  srcSlice: 数据来源切片

  destSlice: 目标切片

举个例子:

  1. func main() {
  2. // copy()复制切片
  3. a := []int{, , , , }
  4. c := make([]int, , )
  5. copy(c, a) //使用copy()函数将切片a中的元素复制到切片c
  6. fmt.Println(a) //[1 2 3 4 5]
  7. fmt.Println(c) //[1 2 3 4 5]
  8. c[] =
  9. fmt.Println(a) //[1 2 3 4 5]
  10. fmt.Println(c) //[1000 2 3 4 5]
  11. }

从切片中删除元素

Go语言中并没有删除切片元素的专用方法,我们可以使用切片本身的特性来删除元素。 代码如下:

  1. func main() {
  2. // 从切片中删除元素
  3. a := []int{, , , , , , , }
  4. // 要删除索引为2的元素
  5. a = append(a[:], a[:]...)
  6. fmt.Println(a) //[30 31 33 34 35 36 37]
  7. }

总结一下就是:要从切片a中删除索引为index的元素,操作方法是a = append(a[:index], a[index+1:]...)

练习题

1.请写出下面代码的输出结果。

  1. func main() {
  2. var a = make([]string, , )
  3. for i := ; i < ; i++ {
  4. a = append(a, fmt.Sprintf("%v", i))
  5. }
  6. fmt.Println(a)
  7. }

2.请使用内置的sort包对数组var a = [...]int{3, 7, 8, 9, 1}进行排序。

Go语言 ( 切片)的更多相关文章

  1. Go 语言切片(Slice)

    Go 语言切片是对数组的抽象. Go 数组的长度不可改变,在特定场景中这样的集合就不太适用,Go中提供了一种灵活,功能强悍的内置类型切片("动态数组"),与数组相比切片的长度是不固 ...

  2. Go语言切片

    切片 Go 语言切片相当于是对数组的抽象. 由于Go 数组的长度不可改变,在特定场景中这样的集合就不太适用,Go中提供了一种灵活,功能强悍的内置类型切片("动态数组"),与数组相比 ...

  3. 17 Go Slices: usage and internals GO语言切片: 使用和内部

    Go Slices: usage and internals  GO语言切片: 使用和内部 5 January 2011 Introduction Go's slice type provides a ...

  4. Go 语言 切片的使用(增删改查)

    Go 语言 切片的使用(增删改查) 引言Golang 的数组是固定长度,可以容纳相同数据类型的元素的集合.但是当长度固定了,在使用的时候肯定是会带来一些限制,比如说:申请的长度太大会浪费内存,太小又不 ...

  5. go语言切片切片与指针

    go语言 1.切片的定义 切片不是真正意义上的动态数组,是引用类型. var arraySlice []int

  6. Go语言 切片长度和容量

    package main import "fmt" func main() { s := []int{2, 3, 5, 7, 11, 13} printSlice(s) // Sl ...

  7. go语言切片作为函数参数的研究

    slice作为函数参数是值传递 golang中的切片slice底层通过数组实现,slice类似一个结构体,其中一个字段保存的是底层数组的地址,还有长度(len) 和 容量(cap)两个字段. 结构体作 ...

  8. go语言 切片表达式

    切片表达式 切片的底层就是一个数组,所以我们可以基于数组通过切片表达式得到切片. 切片表达式中的low和high表示一个索引范围(左包含,右不包含),得到的切片长度=high-low,容量等于得到的切 ...

  9. Go语言切片一网打尽,别和Java语法傻傻分不清楚

    前言 我总想着搞清楚,什么样的技术文章才算是好的文章呢?因为写一篇今后自己还愿意阅读的文章并不容易,暂时只能以此为目标努力. 最近开始用Go刷一些题,遇到了一些切片相关的细节问题,这里做一些总结.切片 ...

随机推荐

  1. learning armbian steps(11) ----- armbian 源码分析(六)

    接下来我们来分析一下uboot的编写过程: 从 lib/compilation.sh  89开始阅读: compile_uboot() { # not optimal, but extra clean ...

  2. Xamarin.IOS/Mac开发中遇到的问题

    虚拟机中安装的mac系统无法识别iphone 今天在 Xamarin.iOS 应用的免费预配 时,进行到 5.插入要在其中部署应用的 iOS 设备. 在第8选择iphone设备时,发现iphone并没 ...

  3. Theano入门笔记1:Theano中的Graph Structure

    译自:http://deeplearning.net/software/theano/extending/graphstructures.html#graphstructures 理解Theano计算 ...

  4. 【JZOJ6222】【20190617】可爱

    题目 给定一个长度为\(n\)的串,定义两个串匹配当且仅当两个串长度相同并且不同字符至多一个 对于每一个长度为\(m\)的子串输出和它匹配的子串个数 $1 \le n \le 10^5  ,  m \ ...

  5. gitbook+git+typora 的使用过程

    Typora 下载地址:https://typora.io/ gitbook 第一步:安装 npm install -g gitbook-cli 第二步:使用 对要操作的文件夹执行命令 gitbook ...

  6. 微信小程序 wxs的简单应用

    Demo地址:微信小程序wxs的简单应用 案例分析 张三.李四.王五,各自分别都有数量不等的车,现在需要列表显示名字及他们拥有车的数量, list数据结构如下,当我们使用wx:for进行显示时,发现个 ...

  7. element ui 中的时间选择器怎么设置默认值/el-date-picker区间选择器怎么这是默认值

    template代码 <el-date-picker value-format="yyyy-MM-dd" v-model="search.date" ty ...

  8. Mybatis 面试题

    题目: 什么是Mybatis?  Mybatis27题 Mybaits的优点 Mybatis27题 MyBatis框架的缺点 Mybatis27题 MyBatis框架适用场合Mybatis27题 My ...

  9. 【软工实践】Beta冲刺(5/5)

    链接部分 队名:女生都队 组长博客: 博客链接 作业博客:博客链接 小组内容 恩泽(组长) 过去两天完成了哪些任务 描述 将数据分析以可视化形式展示出来 新增数据分析展示等功能API 服务器后端部署, ...

  10. android: 结合BottomNavigationView、ViewPager和Fragment 实现左右滑动的效果

    主界面:MainActivity package com.yongdaimi.android.androidapitest; import android.os.Bundle; import andr ...