Go语言程序设计(五)切片

一、切片的定义

        在Go语言中,切片(Slice)是数组的一个引用,它会生成一个指向数组的指针,并通过切片长度关联到底层数组部分或者全部元素。切片还提供了一系列对数组的管理功能(append、copy),可以随时动态扩充存储空间,并且可以被随意传递而不会导致所管理的数组元素被重复复制。根据以上特征,切片通常被用来实现变长数组,而且操作灵活。

切片的数据结构原型定义如下:

src/ pkg/ runtime/ runtime. h

struct Slice
{                   //must not move anything
    byte* array;    //actual data
    unit32 len;     //number of elements
    unit32 cap;     //allocated number of elements
};

由切片数据结构的原型定义可以看到,它抽象为以下三个部分:

  • 指向被引用的底层数组的指针。
  • 切片中元素的个数。
  • 切片分配的存储空间。

二、切片的声明与创建

切片声明与创建的方法有三种:基于底层数组创建、直接创建或使用make()函数创建。

1、基于底层数组创建

在创建切片时,可以基于一个底层数组,切片可以只使用数组的一部分元素或所有元素,甚至可以创建一个比底层数组还要大的数组切片,因为切片可以动态增长。创建切片的格式如下:

var sliceName [ ]dataType

说明:

  1. 切片名的命名规则和变量名相同,遵循标识符命名规则。
  2. 在创建切片时,不要指定切片的长度。
  3. 切片的类型可以是Go语言的任何基本数据类型。

例如:

var slice1 [] int

        上例中定义了一个整型切片slicel,注意不要指定切片长度,如果指定了长度就成了定义数组了。当一个切片定义好以后,如果还没有被初始化,默认值为nil, 而且切片的长度为0。切片的长度可以使用内置函数len()获取,还可以使用内置函数cap()获取切片的内存容量。

所以,当一个切片定义好以后,还要对切片进行初始化,这样切片才可用。对于上例,假如已经定义了一个整型数组array1,则切片slice1的初始化方式如下:

slice1 = array1[start : end]

        从上式可以看到,Go语言支持以array1[start : end]的方式基于一个底层数组来生成切片,即切片引用的数组元素由array1[start]到 array1[end],但不包含array1[end]。

如果要引用底层数组的所有元素,可采取的方式如下:

slicel = array1
slice1 = array1[ :]
slicel = array1[0: len(array1)] 

2、直接创建切片

直接创建切片,即在定义切片的时候初始化切片元素,例如:

var slice1 =[]int{1,2,3,4,5}

3、使用make函数创建切片

内置函数make()用于灵活的创建切片

var slicel = make([]int,5)

上式创建了一个有5个元素的整型切片slicel,元素的初值为0。
在使用make()函数创建切片时,还可以为切片元素预留存储空间。例如:

var slice1 = make([]int, 5,10)

上式表示,创建整型切片slice1,元素个数为5,元素初值为0,并预留10个元素的存储空间。
 

三、切片元素的访问与遍历

        切片元素的遍历和数组元素的遍历一样,要通过元素下标访问,另外也可以使用关键字range遍历所有切片元素。切片元素访问的一般格式如下:

sliceName [sliceIndex]

遍历同数组,使用range关键字表达式,有两个返回值,第一个是元素的下标,第二个是元素的值。

四、切片的操作

1、切片元素的增加

        可以使用append()函数向切片尾部添加新元素,这些元素保存到底层数组。append并不会影响原来切片的属性,它返回变更后新的切片对象。

        与数组相比,除了都有长度(length)以外,切片多了一个容量(capacity)的概念,即切片中元素的个数和分配的存储空间是两个不同的值。如果在增加新元素时超出cap的限制,则底层会重新分配一块“够大”的内存,一般来说是将原来的内存空间扩大二倍,然后再将数据从原来的内存复制到新的内存块。

2、切片元素的复制

        使用切片长时间引用“超大”的底层数组,会导致严重的内存浪费。可以新建一个小的slice对象,然后将所需的数据复制过去。函数copy()可以在切片之间复制元素,能够复制的数量取决于复制方和被复制方的长度值,通常取最小值。需要注意的是,在同一底层数组的不同切片间复制元素时,元素位置会发生重叠。

//切片的复制
package main

import(
    "fmt" 
)

func main() {
    var slicel= []int{1,2,3,4,5,6,7,8,9,10}
    var slice2 = make([ ] int,3, 5)
    var n int
    //只能复制三个元素
    n= copy(slice2,slice1)
    fmt. Println(n, slice2, len( slice2), cap(slice2))
    //slice3和slice1指向同一底层数组
    slice3 : = slice1[3:6]
    //复制后元素重叠
    n= copy(slice3, slice1[1:5])
    fmt.Println(n, slicel, slice3)
}

编译并运行程序输出结果为:

3 [1 2 3] 3 5
3 [1 2 3 2 3 4 7 8 9 10] [2 3 4]

        通过输出结果可以看出,在将slice1复制到slice2时,由于slice2的长度最小为3,所以只能将slice1的前三个元素复制给slice2。而将slicel1复制到slice3 时,由于slicel1和slice3指向同一个底层数组,所以复制后元素重叠。slice3刚创建时,它引用的是底层数组的[4,5,6]三个元素,复制后slice1将[2,3,4]三个元素复制给slice3 ,所以最后slice3 的元素[2,3,4]覆盖了slice1的元素[4,5,6]。

3、排序和搜索切片

        标准库中的sort包提供了对整型、浮点型和字符串类型切片进行排序的函数,检查一个切片是否排序好的函数,以及使用二分搜索算法在一个有序切片中搜索-一个元素的函数。 同时提供了通用sort.Sort ()和sort.Search ()函数,可用于任何自定义的数据。这些函数在表4-2中有列出。

  • 2
    点赞
  • 2
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值