Go语言基础之数组
本文主要介绍Go语言中数组(array)及它的基本使用。
数组是同一类型元素的集合。例如,整数集合 5,8,9,79,76 形成一个数组。Go 语言中不允许混合不同类型的元素,例如包含字符串和整数的数组。(译者注:当然,如果是 interface{} 类型数组,可以包含任意类型)
一、Array(数组)
数组是同一种数据类型元素的集合。 在Go语言中,数组从声明时就确定,使用时可以修改数组成员,但是数组大小不可变化。 基本语法:
// 定义一个长度为3元素类型为int的数组a
var a [3]int
二、数组定义:
var 数组变量名 [元素数量]T
比如:var a [5]int
, 数组的长度必须是常量,并且长度是数组类型的一部分。一旦定义,长度不能变。 [5]int
和[10]int
是不同的类型。
var a [3]int
var b [4]int
a = b //不可以这样做,因为此时a和b是不同的类型
数组可以通过下标进行访问,下标是从0
开始,最后一个元素下标是:len-1
,访问越界(下标在合法范围之外),则触发访问越界,会panic。
三、数组的初始化
数组的初始化也有很多方式。
方法一
初始化数组时可以使用初始化列表来设置数组元素的值。
func main() {
var testArray [3]int //数组会初始化为int类型的零值
var numArray = [3]int{1, 2} //使用指定的初始值完成初始化,其他值为0
var cityArray = [3]string{"北京", "上海", "深圳"} //使用指定的初始值完成初始化
fmt.Println(testArray) //[0 0 0]
fmt.Println(numArray) //[1 2 0]
fmt.Println(cityArray) //[北京 上海 深圳]
}
方法二
按照上面的方法每次都要确保提供的初始值和数组长度一致,一般情况下我们可以让编译器根据初始值的个数自行推断数组的长度,例如:
func main() {
var testArray [3]int
var numArray = [...]int{1, 2}
var cityArray = [...]string{"北京", "上海", "深圳"}
fmt.Println(testArray) //[0 0 0]
fmt.Println(numArray) //[1 2]
fmt.Printf("type of numArray:%T
", numArray) //type of numArray:[2]int
fmt.Println(cityArray) //[北京 上海 深圳]
fmt.Printf("type of cityArray:%T
", cityArray) //type of cityArray:[3]string
}
方法三
我们还可以使用指定索引值的方式来初始化数组,例如:
func main() {
a := [...]int{1: 1, 3: 5}
fmt.Println(a) // [0 1 0 5]
fmt.Printf("type of a:%T
", a) //type of a:[4]int
}
定义并赋值
// 三种方式
var a [4]int=[4]int{1,2,3,4}
var a =[4]int{1,2,3,4}
a :=[4]int{1,2,3,4}
fmt.Println(a)
// 定义并赋初值(其他方式)
var a [4]int = [4]int{1,2}
fmt.Println(a)
var b [4]int = [4]int{2:5}
fmt.Println(b)
var c [4]int = [4]int {3:4}
fmt.Println(c)
var d [4]int =[4]int{3:4,2:3}
fmt.Println(d)
// 数组长度
a := [4]int{1,2,3}
fmt.Println(len(a))
数组大小固定(大小一旦定义,不能改变),数组的大小是类型的一部分
四、数组的遍历
遍历数组a有以下两种方法:
func main() {
var a = [...]string{"北京", "上海", "深圳"}
// 方法1:for循环遍历
for i := 0; i < len(a); i++ {
fmt.Println(a[i])
}
// 方法2:for range遍历 (range:关键字,不是内置函数)把数组放在range关键字后
for index, value := range a {
fmt.Println(index, value)
}
for _, value := range a {
fmt.Println(value)
}
}
五、多维数组
Go语言是支持多维数组的,我们这里以二维数组为例(数组中又嵌套数组)。
二维数组的定义
func main() {
a := [3][2]string{
{"北京", "上海"},
{"广州", "深圳"},
{"成都", "重庆"},
}
fmt.Println(a) //[[北京 上海] [广州 深圳] [成都 重庆]]
fmt.Println(a[2][1]) //支持索引取值:重庆
}
二维数组的遍历
func main() {
a := [3][2]string{
{"北京", "上海"},
{"广州", "深圳"},
{"成都", "重庆"},
}
for _, v1 := range a {
for _, v2 := range v1 {
fmt.Printf("%s ", v2)
}
fmt.Println()
}
}
输出:
北京 上海
广州 深圳
成都 重庆
不确定维度
注意: 多维数组只有第一层可以使用...
来让编译器推导数组长度。例如:
//支持的写法
a := [...][2]string{
{"北京", "上海"},
{"广州", "深圳"},
{"成都", "重庆"},
}
//不支持多维数组的内层使用...
b := [3][...]string{
{"北京", "上海"},
{"广州", "深圳"},
{"成都", "重庆"},
}
六、数组是值类型
数组是值类型,赋值和传参会复制整个数组。因此改变副本的值,不会改变本身的值。
func modifyArray(x [3]int) {
x[0] = 100
}
func modifyArray2(x [3][2]int) {
x[2][0] = 100
}
func main() {
// go中数组是值类型,当参数传递时,copy一份传过去,值被copy是相当于复制了一份传过去了
a := [3]int{10, 20, 30}
modifyArray(a) //在modify中修改的是a的副本x
fmt.Println(a) //[10 20 30]
b := [3][2]int{
{1, 1},
{1, 1},
{1, 1},
}
modifyArray2(b) //在modify中修改的是b的副本x
fmt.Println(b) //[[1 1] [1 1] [1 1]]
}
/*
不准确(因为py中一切皆对象,不可变数据类型也是对象,对象就是地址)
py中不可变数据类型(值类型)当做参数传递到函数中,如果在函数中改变该值,原值不会改变
py中可变数据类型(引用类型:地址)当做参数传递到函数中,如果在函数中改变该值,原值会改变
*/
注意:
- 数组支持 “==“、”!=” 操作符,因为内存总是被初始化过的。
[n]*T
表示指针数组,*[n]T
表示数组指针 。
八、总结
一维数组:
- 数组定义方式一:
var 数组变量名 [元素数量]T``var arr[4]int
默认值为T类型的零值 - 数组定义方式二: var 数组变量名= [元素数量]T{值1, 值2}
- 数组定义方式三:var 数组变量名=[...]T{1: 1, 3: 5} 使用指定的初始值完成初始化,其他值为零值,int默认值为 0, string默认值为空字符串
- 三种定义和赋值方式,数组大小固定(大小一旦定义,不能改变),数组的大小是类型的一部分;
- 其他赋值方式;
- 数组是值类型,保存的是值
- 数组的长度len,循环数组range,
- 函数传参不会被改变
多维数组:
- 二维数据定义的方式
- 同一维数组基本一样