阅读目录
一、数组
数组的声明和初始化
在 Go 语言中,数组是固定长度的、同一类型的数据集合。数组中包含的每个数据项被称为数组元素,一个数组包含的元素个数被称为数组的长度。
在 Go 语言中,你可以通过 []
来标识数组类型,但需要指定长度和元素类型,使用时可以修改数组成员,但是数组大小不可变化。以下是一些常见的数组声明方法:
1 2 3 4 5 |
|
和普通变量赋值一样,数组也可以通过 :=
进行一次性声明和初始化,所有数组元素通过 {}
包裹,然后通过逗号分隔多个元素
1 |
|
语法糖省略数组长度的声明
1 |
|
数组在初始化的时候,如果没有填满,则空位会通过对应的元素类型零值填充
1 2 3 4 5 |
|
我们还可以初始化指定下标位置的元素值,未设置的位置也会以对应元素类型的零值填充
1 2 3 4 |
|
数组的长度是该数组类型的一个内置常量,可以用 Go 语言的内置函数 len()
来获取
1 |
|
数组元素的访问和设置
可以使用数组下标来访问 Go 数组中的元素,数组下标默认从 0 开始,len(arr)-1
表示最后一个元素的下标:
1 2 |
|
遍历数组
遍历数组a有以下两种方法
1 2 3 4 5 6 7 8 9 10 11 12 |
|
如果我们不想获取索引值,可以这么做:
1 2 3 |
|
如果只想获取索引值,可以这么做:
1 2 3 |
|
多维数组
这里以二维数组为例(数组中又嵌套数组)
二维数组的定义
1 2 3 4 5 6 7 8 9 |
|
二维数组的遍历
1 2 3 4 5 6 7 8 9 10 11 12 13 |
|
注意: 多维数组只有第一层可以使用...
来让编译器推导数组长度。例如:
1 2 3 4 5 6 7 8 9 10 11 12 |
|
数组是值类型
数组是值类型,赋值和传参会复制整个数组。因此改变副本的值,不会改变本身的值。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 |
|
数组类型的不足
由于数组类型变量一旦声明后长度就固定了,这意味着我们将不能动态添加元素到数组,如果要这么做的话,需要先创建一个容量更大的数组,然后把老数组的元素都拷贝过来,最后再添加新的元素,如果数组尺寸很大的话,势必会影响程序性能,例如
1 2 3 4 5 6 7 |
|
另外,数组是值类型,这意味着作为参数传递到函数时,传递的是数组的值拷贝,也就是说,会先将数组拷贝给形参,然后在函数体中引用的是形参而不是原来的数组,当我们在函数中对数组元素进行修改时,并不会影响原来的数组,
这种机制带来的另一个负面影响是当数组很大时,值拷贝会降低程序性能。综合以上因素,我们迫切需要一个引用类型的、支持动态添加元素的新「数组」类型,这就是下篇教程将要介绍的切片类型,实际上,我们在 Go 语言中很少使用数组,大多数时候会使用切片取代它。
二、切片
在 Go 语言中,切片是一个新的数据类型,与数组最大的不同在于,切片的类型字面量中只有元素的类型,没有长度
切片定义
声明切片类型的基本语法如下
1 2 3 4 |
|
创建切片的方法主要有三种 —— 基于数组、切片和直接创建
基于数组
切片可以基于一个已存在的数组创建。从这个层面来说,数组可以看作是切片的底层数组,而切片则可以看作是数组某个连续片段的引用。切片可以只使用数组的一部分元素或者整个数组来创建,甚至可以创建一个比所基于的数组还要大的切片:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 |
|
Go 语言支持通过 array[start:end]
这样的方式基于数组生成一个切片,start
表示切片在数组中的下标起点,end
表示切片在数组中的下标终点,两者之间的元素就是切片初始化后的元素集合,通过上面的示例可以看到,和字符串切片一样,这也是个左闭右开的集合,下面几种用法也都是合法的:
1 2 3 4 5 6 7 8 |
|
基于切片
1 2 3 4 5 |
|
使用make()函数构造切片
1 2 3 4 5 |
|
示例:
1 2 3 4 5 6 7 8 |
|
判断切片是否为空
要检查切片是否为空,请始终使用len(s) == 0
来判断,而不应该使用s == nil
来判断。
切片的赋值拷贝
下面的代码中演示了拷贝前后两个变量共享底层数组,对一个切片的修改会影响另一个切片的内容,这点需要特别注意
1 2 3 4 5 6 7 |
|
切片遍历
切片的遍历方式和数组是一致的,支持索引遍历和for range
遍历
1 2 3 4 5 6 7 8 9 10 11 |
|
动态增加元素
Go语言的内建函数append()
可以为切片动态添加元素。 可以一次添加一个元素,可以添加多个元素,也可以添加另一个切片中的元素(后面加…)
1 2 3 4 5 6 7 8 9 10 |
|
每个切片会指向一个底层数组,这个数组的容量够用就添加新增元素。当底层数组不能容纳新增的元素时,切片就会自动按照一定的策略进行“扩容”,此时该切片指向的底层数组就会更换。“扩容”操作往往发生在append()
函数调用时,所以我们通常都需要用原变量接收append函数的返回值。
举个例子:
1 2 3 4 5 6 7 8 |
|
append()函数还支持一次性追加多个元素。 例如:
1 2 3 4 5 6 7 8 9 |
|
切片的扩容策略
可以通过查看$GOROOT/src/runtime/slice.go
源码,其中扩容相关代码如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 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)。
需要注意的是,切片扩容还会根据切片中元素的类型不同而做不同的处理,比如int
和string
类型的处理方式就不一样。
使用copy()函数复制切片
首先我们来看一个问题:
1 2 3 4 5 6 7 8 9 |
|
由于切片是引用类型,所以a和b其实都指向了同一块内存地址。修改b的同时a的值也会发生变化。
Go语言内建的copy()
函数可以迅速地将一个切片的数据复制到另外一个切片空间中,copy()
函数的使用格式如下
1 2 3 4 |
|
示例:
1 2 3 4 5 6 7 8 9 10 11 |
|
从切片中删除元素
Go语言中并没有删除切片元素的专用方法,我们可以使用切片本身的特性来删除元素
1 2 3 4 5 6 7 8 |
|