容器(一)——数组和切片

5 篇文章 0 订阅

容器:存储和组织数据的方式

3.1 数组—固定大小的连续空间

数组是一段固定长度的连续内存区域。

在Go语言中,数组从声明时就确定,使用时可修改数组成员,但数组大小不可变化

3.1.1 声明数组

数组的写法如下:

var 数组变量名 [元素数量]T

其中

  • 数组变量名: 数组声明及使用时的变量名。

  • 元素数量: 数组的元素数量。可以是一个表达式,但最终通过编译期计算的结果必须是整型数值。也就是说,元素数量不能含有到运行时才能确认大小的数值

  • T 可以是任意基本类型,包括 T为数组本身。但类型为数组本身时,可以实现多维数组。

数组演示例子:

var sports [3]string
sports[0] = "pingpang"
sports[1] = "basketball"
sports[2] = "football"
fmt.Println(sports)

3.1.2 初始化数组

数组可以在声明时使用初始化列表进行元素设置,参考下面的代码:

var test = [3]int{1,2,3}

这种方式编写时,需要保证大括号后面的元素数量与数组的大小一致。但一般情况下,这个过程可 以交给编译器,让编译器在编译时,根据元素个数确定数组大小。

var test = [...]int{1,3,4,5,6}

“… ”表示让编译器确定数组大小, 以上例子里编译器会自动为这个数组设置元素个数为5

3.1.3 遍历数组

遍历数据和遍历切片类似,如以下代码:

var sports [3]string
sports[0] = "pingpang"
sports[1] = "basketball"
sports[2] = "football"
for k, v := range sports {
	fmt.Println(k,v)
}

3.2 切片—动态分配大小的连续空间

Go 语言切片的内部结构包含地址、 大小和容量。 切片一般用于快速地操作一块数据集合。 如果将数据集合 比作切糕的话, 切片就是你要的“那一块”。

3.2.1 从数组或切片生成新的切片

切片默认指向一段连续内存区域,可以是数组,也可以是切片本身。

从连续内存区域生成切片是常见的操作,格式如下:

slice [开始位置:结束位置]

  • slice 表示目标切片对象。

  • 开始位置对应目标切片对象的索引。

  • 结束位置对应目标切片的结束索引。

从数组或切片生成新的切片拥有如下特性。

  • 取出的元素数量为:结束位置-开始位置。
  • 取出元素不包含结束位置对应的索引,切片最后一个元素使用 slice [len(slice]获取。
  • 当缺省开始位置时,表示从连续区域开头到结束位置。
  • 当缺省结束位置时,表示从开始位置到整个连续区域末尾。
  • 两者同时缺省时,与切片本身等效。
  • 两者同时为空时,等效于空切片,一般用于切片复位。
  • 根据索引位置取切片slice 元素值时,取值范围是(0~ len(slice)-1 ),超界会报运行时错误。生成切片时,结束位置可以填写len(slice)但不会报错。
1. 从指定范围生成切片

从指定范围生成切片示例代码如下:

var highBuilding [30]int

for i:=0; i < 30; i++ {
    highBuilding[i] = i + 1
}
//区间
fmt.Println(highBuilding[10:15])

//中间到尾部的所有元素
fmt.Println(highBuilding[20:])

//开头到中间的所有元素
fmt.Println(highBuilding[:20])

切片有点像C语言里的指针。指针可以做运算,但代价是内存操作越界。切片在指针基础上增加了大小,约束了切片对应的内存区域,切片使用中无法对切片内部的地址和大小进行手动调整,因此切片比指针更安全、 强大

2. 表示原有的切片

生成切片格式中,当开始和结束的范围都被忽略,则生成的切片表示和原切片一致的切片,并切数据内容一致,代码如下:

a :=[]int{1,2,3}
fmt.Println(a[:])
3. 重置切片,清空元素

把切片的开始和结束位置都设为0,生产的切片将变空,代码如下:

 a :=[]int{1,2,3}
 fmt.Println(a[0:0])

3.2.2 声明切片

每一种类型都可以拥有其切片类型,表示多个类型元素的连续集合。因此切片类型可以被声明。切片类型声明格式如下:

var name []T

  • name表示切片类型的变量名
  • T表示切片类型对应的元素类型

下面的代码展示了切片声明的使用过程。

var strList []string //声明字符串切片
var numList []int	
var numListEmpty = []int{}	//声明一个空切片

//输出3个切片
fmt.Println(strList, numList, numListEmpty)
//切片判定空的结果
fmt.Println(strList == nil)			//结果:true
fmt.Println(numList == nil)			//结果:true
fmt.Println(numListEmpty == nil)	//结果:false

numListEmpty 已经被分配到了内存,但没有元素,因此和 nil 比较时是false,切片是动态结构,只能与nil判定相等,不能互相判等时

3.2.3 使用make()函数构造切片

如果需要动态地创建 个切片,可以使用 make()内建函数,格式如下:

make ( []T, size, cap )

  • T: 切片的元素类型

  • size:就是为这个类型分配多少个元素。

  • cap:预分配的元素数 ,这个值设定后不影响 size,只是能提前分配空间,降低多次分配空间造成的性能问题。

示例如下:

a := make([]int, 2)
b := make([]int, 2, 10)

fmt.Println(a, b, len(a), len(b))

3.2.4 使用append()函数为切片添加元素

代码如下:

var nums []int

for i:=0; i < 10; i++ {
	nums = append(nums, i)
	fmt.Printf("len:%d cap:%d pointer:%p\n", len(nums), cap(nums), nums)
}

3.2.5 复制切片元素到另一个切片

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

copy(destSlice, srcSlice[]T) int

  • srcSlice为数据来源切片
  • destSlice为复制的目标。目标切片必须分配过空间且足够承载复制的元素个数。来源和目标的类型一致,copy的返回值表示实际发生复制的元素个数。
slice1 := []int{1, 2, 3, 4, 5}
slice2 := []int{5, 4, 3}
copy(slice2, slice1) // 只会复制slice1的前3个元素到slice2中
copy(slice1, slice2) // 只会复制slice2的3个元素到slice1的前3个位置

3.2.6 从切片中删除元素

Go语言并没有对删除切片元素提供专用的语法或者接口,需要使用切片本身的特性来删除元素。示例代码如下:

seq := []string{"a", "b", "c", "d", "e"}
index := 2
//查看删除位置之前的元素和之后的元素
fmt.Println(seq[:index], seq[index+1:])
//将删除点前后的元素连接起来
seq = append(seq[:index], seq[index+1:]...)
fmt.Println(seq)

Go 中切片删除元元素的本质 :以被删除元素为分界点,将前后两个部分的内存重新连接起来

提示:连续容器的元素删除无论是在任何语言中,都要将删除点前后的元素移动到新的位置。随着元素的增加,这个过程将会变得极为耗时 因此,当业务需要大量、频繁地从一个切片中删除元素时,如果对性能要求较高,就需要反思是否需要更换其他的容器(如双链表等能快速从删除点删除元素)。

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Golang的数组切片是两种不同的数据类型,用于存储相同数据类型的容器数组的长度是固定的,而切片的长度是可变的。在日常应用中,切片的使用更为普遍。 数组在声明时需要指定长度,并且在初始化时必须提供相同长度的元素。例如,`a := int{1, 2, 3}`就是一个长度为3的整数数组数组的长度一旦确定后就不能更改。 切片是基于数组的引用类型。它不需要指定固定的长度,并且可以根据需要动态扩展或缩小。切片包装着底层数组,通过指定起始索引和结束索引来指定子集。例如,`b := a[:]`就是一个切片,它包含了数组a的所有元素。 数组适用于需要固定长度的场景,而切片适用于长度可变的情况。在实际应用中,切片更常用,因为它提供了更大的灵活性和便利性。 总结: - 数组是长度固定的容器切片是长度可变的容器; - 数组在声明时需要指定长度,切片则不需要; - 数组的长度一旦确定后就不能更改,而切片可以根据需要动态扩展或缩小; - 切片是基于数组的引用类型,可以通过指定起始索引和结束索引来指定子集。 参考资料: Golang中的「数组」和「切片」都是存储同一数据类型的容器,只不过Golang中的数组长度是固定的,而切片的长度是可变化的。 切片是引用类型,切片包装的数组称为该切片的底层数组。我们来看一段代码://a是一个数组,注意数组是一个固定长度的,初始化时候必须要指定长度,不指定长度的话就是切片了 a := int{1, 2, 3} //b是数组,是a...。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值