Go语言 ( 切片)
本文主要介绍Go语言中切片(slice)及它的基本使用。
引子
因为数组的长度是固定的并且数组长度属于类型的一部分,所以数组有很多的局限性。 例如:
- func arraySum(x []int) int{
- sum :=
- for _, v := range x{
- sum = sum + v
- }
- return sum
- }
这个求和函数只能接受[3]int
类型,其他的都不支持。 再比如,
- a := []int{, , }
数组a中已经有三个元素了, 我们不能再及所需往数组a中添加新元素了。
切片
切片(Slice)是一个拥有相同类型元素的可变长度的序列。它是基于数组类型做的一层封装。它非常灵活,支持自动扩容。
切片是一个引用类型,它的内部结构包含地址
、长度
和容量
。切片一般用于快速地操作一块数据集合。
切片的定义
声明切片类型的基本语法如下:
- var name []T
其中:
name: 表示变量名
T: 表示切片中的元素类型
举个例子:
- func main() {
- // 声明切片类型
- var a []string //声明一个字符串切片
- var b = []int{} //声明一个整型切片并初始化
- var c = []bool{false, true} //声明一个布尔切片并初始化
- var d = []bool{false, true} //声明一个布尔切片并初始化
- fmt.Println(a) //[]
- fmt.Println(b) //[]
- fmt.Println(c) //[false true]
- fmt.Println(a == nil) //true
- fmt.Println(b == nil) //false
- fmt.Println(c == nil) //false
- // fmt.Println(c == d) //切片是引用类型,不支持直接比较,只能和nil比较
- }
切片的长度和容量
切片拥有自己的长度和容量,我们可以通过使用内置的len()函数求长度,使用内置的cap()函数求切片的容量。
基于数组定义切片
由于切片的底层就是一个数组,所以我们可以基于数组定义切片。
- func main() {
- // 基于数组定义切片
- a := []int{, , , , }
- b := a[:] //基于数组a创建切片,包括元素a[1],a[2],a[3] 包头不包尾
- fmt.Println(b) //[56 57 58]
- fmt.Printf("type of b:%T\n", b) //type of b:[]int
- }
还支持如下方式:
- c := a[:] //[56 57 58 59]
- d := a[:] //[55 56 57]
- e := a[:] //[55 56 57 58 59]
切片再切片
除了基于数组得到切片,我们还可以通过切片来得到切片。
- func main() {
- //切片再切片
- a := [...]string{"北京", "上海", "广州", "深圳", "成都", "重庆"}
- fmt.Printf("a:%v type:%T len:%d cap:%d\n", a, a, len(a), cap(a))
- b := a[:]
- fmt.Printf("b:%v type:%T len:%d cap:%d\n", b, b, len(b), cap(b))
- c := b[:]
- fmt.Printf("c:%v type:%T len:%d cap:%d\n", c, c, len(c), cap(c))
- }
输出:
- a:[北京 上海 广州 深圳 成都 重庆] type:[]string len: cap:
- b:[上海 广州] type:[]string len: cap:
- c:[广州 深圳 成都 重庆] type:[]string len: cap:
注意: 对切片进行再切片时,索引不能超过原数组的长度,否则会出现索引越界的错误。
使用make()函数构造切片
我们上面都是基于数组来创建的切片,如果需要动态的创建一个切片,我们就需要使用内置的make()
函数,格式如下:
- make([]T, size, cap)
其中:
T: 切片的元素类型
size:切片中元素的数量
cap: 切片的容量
举个例子:
- func main() {
- a := make([]int, , )
- fmt.Println(a) //[0 0]
- fmt.Println(len(a)) //
- fmt.Println(cap(a)) //
- }
上面代码中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
,例如下面的示例:
- var s1 []int //len(s1)=0;cap(s1)=0;s1==nil
- s2 := []int{} //len(s2)=0;cap(s2)=0;s2!=nil
- s3 := make([]int, ) //len(s3)=0;cap(s3)=0;s3!=nil
所以要判断一个切片是否是空的,要是用len(s) == 0
来判断,不应该使用s == nil
来判断。
切片的赋值拷贝
下面的代码中演示了拷贝前后两个变量共享底层数组,对一个切片的修改会影响另一个切片的内容,这点需要特别注意。
- func main() {
- s1 := make([]int, ) //[0 0 0]
- s2 := s1 // 将s1直接赋值给s2,s1和s2 共用一个底层数组
- s2[] =
- fmt.Println(s1) // [100 0 0]
- fmt.Println(s2) // [100 0 0]
- }
切片遍历
切片的遍历方式和数组是一致的,支持索引遍历和for range
遍历。
- func main() {
- s := []int{, , }
- for i := ; i < len(s); i++ {
- fmt.Println(i, s[i])
- }
- for index, value := range s {
- fmt.Println(index, value)
- }
- }
append()方法为切片添加元素
Go语言的内建函数append()
可以为切片动态添加元素。 每个切片会指向一个底层数组,这个数组能容纳一定数量的元素。当底层数组不能容纳新增的元素时,切片就会自动按照一定的策略进行“扩容”,此时该切片指向的底层数组就会更换。“扩容”操作往往发生在append()
函数调用时。 举个例子:
- func main() {
- //append()添加元素和切片扩容
- var numSlice []int
- for i := ; i < ; i++ {
- numSlice = append(numSlice, i)
- fmt.Printf("%v len:%d cap:%d ptr:%p\n", numSlice, len(numSlice), cap(numSlice), numSlice)
- }
- }
输出:
- [] len: cap: ptr:0xc0000a8000
- [ ] len: cap: ptr:0xc0000a8040
- [ ] len: cap: ptr:0xc0000b2020
- [ ] len: cap: ptr:0xc0000b2020
- [ ] len: cap: ptr:0xc0000b6000
- [ ] len: cap: ptr:0xc0000b6000
- [ ] len: cap: ptr:0xc0000b6000
- [ ] len: cap: ptr:0xc0000b6000
- [ ] len: cap: ptr:0xc0000b8000
- [ ] len: cap: ptr:0xc0000b8000
从上面的结果可以看出:
append()
函数将元素追加到切片的最后并返回该切片。- 切片numSlice的容量按照1,2,4,8,16这样的规则自动进行扩容,每次扩容后都是扩容前的2倍。
append()函数还支持一次性追加多个元素。 例如:
- var citySlice []string
- // 追加一个元素
- citySlice = append(citySlice, "北京")
- // 追加多个元素
- citySlice = append(citySlice, "上海", "广州", "深圳")
- // 追加切片
- a := []string{"成都", "重庆"}
- citySlice = append(citySlice, a...)
- fmt.Println(citySlice) //[北京 上海 广州 深圳 成都 重庆]
切片的扩容策略
可以通过查看$GOROOT/src/runtime/slice.go
源码,其中扩容相关代码如下:
- newcap := old.cap
- doublecap := newcap + newcap
- if cap > doublecap {
- newcap = cap
- }else {
- if old.len < {
- newcap = doublecap
- }else{
- // Check 0 < newcap to detect overflow
- // and prevent an infinite loop.
- for < newcap && newcap < cap {
- newcap += newcap /
- }
- // Set newcap to the requested cap when
- // the newcap calculation overflowed.
- if newcap <= {
- newcap = cap
- }
- }
- }
从上面的代码可以看出以下内容:
- 首先判断,如果新申请容量(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)。
需要注意的是,切片扩容还会根据切片中元素的类型不同而做不同的处理,比如int
和string
类型的处理方式就不一样。
使用copy()函数复制切片
首先我们来看一个问题:
- func main() {
- a := []int{, , , , }
- b := a
- fmt.Println(a) //[1 2 3 4 5]
- fmt.Println(b) //[1 2 3 4 5]
- b[] =
- fmt.Println(a) //[1000 2 3 4 5]
- fmt.Println(b) //[1000 2 3 4 5]
- }
由于切片是引用类型,所以a和b其实都指向了同一块内存地址。修改b的同时a的值也会发生变化。
Go语言内建的copy()
函数可以迅速地将一个切片的数据复制到另外一个切片空间中,copy()
函数的使用格式如下:
- copy(destSlice, srcSlice []T)
其中:
srcSlice: 数据来源切片
destSlice: 目标切片
举个例子:
- func main() {
- // copy()复制切片
- a := []int{, , , , }
- c := make([]int, , )
- copy(c, a) //使用copy()函数将切片a中的元素复制到切片c
- fmt.Println(a) //[1 2 3 4 5]
- fmt.Println(c) //[1 2 3 4 5]
- c[] =
- fmt.Println(a) //[1 2 3 4 5]
- fmt.Println(c) //[1000 2 3 4 5]
- }
从切片中删除元素
Go语言中并没有删除切片元素的专用方法,我们可以使用切片本身的特性来删除元素。 代码如下:
- func main() {
- // 从切片中删除元素
- a := []int{, , , , , , , }
- // 要删除索引为2的元素
- a = append(a[:], a[:]...)
- fmt.Println(a) //[30 31 33 34 35 36 37]
- }
总结一下就是:要从切片a中删除索引为index
的元素,操作方法是a = append(a[:index], a[index+1:]...)
练习题
1.请写出下面代码的输出结果。
- func main() {
- var a = make([]string, , )
- for i := ; i < ; i++ {
- a = append(a, fmt.Sprintf("%v", i))
- }
- fmt.Println(a)
- }
2.请使用内置的sort
包对数组var a = [...]int{3, 7, 8, 9, 1}
进行排序。
Go语言 ( 切片)的更多相关文章
- Go 语言切片(Slice)
Go 语言切片是对数组的抽象. Go 数组的长度不可改变,在特定场景中这样的集合就不太适用,Go中提供了一种灵活,功能强悍的内置类型切片("动态数组"),与数组相比切片的长度是不固 ...
- Go语言切片
切片 Go 语言切片相当于是对数组的抽象. 由于Go 数组的长度不可改变,在特定场景中这样的集合就不太适用,Go中提供了一种灵活,功能强悍的内置类型切片("动态数组"),与数组相比 ...
- 17 Go Slices: usage and internals GO语言切片: 使用和内部
Go Slices: usage and internals GO语言切片: 使用和内部 5 January 2011 Introduction Go's slice type provides a ...
- Go 语言 切片的使用(增删改查)
Go 语言 切片的使用(增删改查) 引言Golang 的数组是固定长度,可以容纳相同数据类型的元素的集合.但是当长度固定了,在使用的时候肯定是会带来一些限制,比如说:申请的长度太大会浪费内存,太小又不 ...
- go语言切片切片与指针
go语言 1.切片的定义 切片不是真正意义上的动态数组,是引用类型. var arraySlice []int
- Go语言 切片长度和容量
package main import "fmt" func main() { s := []int{2, 3, 5, 7, 11, 13} printSlice(s) // Sl ...
- go语言切片作为函数参数的研究
slice作为函数参数是值传递 golang中的切片slice底层通过数组实现,slice类似一个结构体,其中一个字段保存的是底层数组的地址,还有长度(len) 和 容量(cap)两个字段. 结构体作 ...
- go语言 切片表达式
切片表达式 切片的底层就是一个数组,所以我们可以基于数组通过切片表达式得到切片. 切片表达式中的low和high表示一个索引范围(左包含,右不包含),得到的切片长度=high-low,容量等于得到的切 ...
- Go语言切片一网打尽,别和Java语法傻傻分不清楚
前言 我总想着搞清楚,什么样的技术文章才算是好的文章呢?因为写一篇今后自己还愿意阅读的文章并不容易,暂时只能以此为目标努力. 最近开始用Go刷一些题,遇到了一些切片相关的细节问题,这里做一些总结.切片 ...
随机推荐
- learning armbian steps(11) ----- armbian 源码分析(六)
接下来我们来分析一下uboot的编写过程: 从 lib/compilation.sh 89开始阅读: compile_uboot() { # not optimal, but extra clean ...
- Xamarin.IOS/Mac开发中遇到的问题
虚拟机中安装的mac系统无法识别iphone 今天在 Xamarin.iOS 应用的免费预配 时,进行到 5.插入要在其中部署应用的 iOS 设备. 在第8选择iphone设备时,发现iphone并没 ...
- Theano入门笔记1:Theano中的Graph Structure
译自:http://deeplearning.net/software/theano/extending/graphstructures.html#graphstructures 理解Theano计算 ...
- 【JZOJ6222】【20190617】可爱
题目 给定一个长度为\(n\)的串,定义两个串匹配当且仅当两个串长度相同并且不同字符至多一个 对于每一个长度为\(m\)的子串输出和它匹配的子串个数 $1 \le n \le 10^5 , m \ ...
- gitbook+git+typora 的使用过程
Typora 下载地址:https://typora.io/ gitbook 第一步:安装 npm install -g gitbook-cli 第二步:使用 对要操作的文件夹执行命令 gitbook ...
- 微信小程序 wxs的简单应用
Demo地址:微信小程序wxs的简单应用 案例分析 张三.李四.王五,各自分别都有数量不等的车,现在需要列表显示名字及他们拥有车的数量, list数据结构如下,当我们使用wx:for进行显示时,发现个 ...
- element ui 中的时间选择器怎么设置默认值/el-date-picker区间选择器怎么这是默认值
template代码 <el-date-picker value-format="yyyy-MM-dd" v-model="search.date" ty ...
- Mybatis 面试题
题目: 什么是Mybatis? Mybatis27题 Mybaits的优点 Mybatis27题 MyBatis框架的缺点 Mybatis27题 MyBatis框架适用场合Mybatis27题 My ...
- 【软工实践】Beta冲刺(5/5)
链接部分 队名:女生都队 组长博客: 博客链接 作业博客:博客链接 小组内容 恩泽(组长) 过去两天完成了哪些任务 描述 将数据分析以可视化形式展示出来 新增数据分析展示等功能API 服务器后端部署, ...
- android: 结合BottomNavigationView、ViewPager和Fragment 实现左右滑动的效果
主界面:MainActivity package com.yongdaimi.android.androidapitest; import android.os.Bundle; import andr ...