|
| 1 | +11 - 数组和切片 |
| 2 | +======================== |
| 3 | + |
| 4 | +上一节:[第十篇 if else 语句](/docs/golang_tutorial_10.md) |
| 5 | +下一节:[第十二篇 包](/docs/golang_tutorial_12.md) |
| 6 | + |
| 7 | +这是本Golang系列教程的第11篇。 |
| 8 | + |
| 9 | +## 数组 |
| 10 | + |
| 11 | +数组是类型相同的元素的集合。例如,整数 5, 8, 9, 79, 76 的集合就构成了一个数组。Go不允许在数组中混合使用不同类型的元素(比如整数和字符串)。 |
| 12 | + |
| 13 | +## 申明 |
| 14 | + |
| 15 | +数组的类型为` n[T]`,其中 `n` 表示数组中元素的个数,`T` 表示数组中元素的类型。元素的个数 `n`也是数组类型的一部分(我们将在稍后详细讨论)。 |
| 16 | + |
| 17 | +有很多声明数组的方式,让我们一个一个地介绍。 |
| 18 | + |
| 19 | +```golang |
| 20 | +package main |
| 21 | + |
| 22 | +import ( |
| 23 | + "fmt" |
| 24 | +) |
| 25 | + |
| 26 | +func main() { |
| 27 | + var a [3]int //int array with length 3 |
| 28 | + fmt.Println(a) |
| 29 | +} |
| 30 | +``` |
| 31 | + |
| 32 | +`var a [3]int `声明了一个长度为 3 的整型数组。数组中的所有元素都被自动赋值为元素类型的 0 值。比如这里 a 是一个整型数组,因此 a 中的所有元素都被赋值为 0(即整型的 0 值)。运行上面的程序,输出为:`[0 0 0]`。 |
| 33 | + |
| 34 | +数组的索引从 0 开始到 `length - 1` 结束。下面让我们给上面的数组赋一些值。 |
| 35 | + |
| 36 | +```golang |
| 37 | +package main |
| 38 | + |
| 39 | +import ( |
| 40 | + "fmt" |
| 41 | +) |
| 42 | + |
| 43 | +func main() { |
| 44 | + var a [3]int //int array with length 3 |
| 45 | + a[0] = 12 // array index starts at 0 |
| 46 | + a[1] = 78 |
| 47 | + a[2] = 50 |
| 48 | + fmt.Println(a) |
| 49 | +} |
| 50 | +``` |
| 51 | + |
| 52 | +`a[0]`表示数组中的第一个元素。程序的输出为:`[12 78 50]`。 |
| 53 | + |
| 54 | +(译者注:可以用下标运算符(`[]`)来访问数组中的元素,下标从 0 开始,例如 `a[0]` 表示数组 a 的第一个元素,`a[1] `表示数组 a 的第二元素,以此类推。) |
| 55 | + |
| 56 | +可以利用**速记声明(shorthand declaration)**的方式来创建同样的数组: |
| 57 | + |
| 58 | +```golang |
| 59 | +package main |
| 60 | + |
| 61 | +import ( |
| 62 | + "fmt" |
| 63 | +) |
| 64 | + |
| 65 | +func main() { |
| 66 | + a := [3]int{12, 78, 50} // shorthand declaration to create array |
| 67 | + fmt.Println(a) |
| 68 | +} |
| 69 | +``` |
| 70 | + |
| 71 | +上面的程序输出为:`[12 78 50]`。 |
| 72 | + |
| 73 | +(译者注:这个例子给出了速记声明的方式:在数组类型后面加一对大括号(`{}`),在大括号里面写元素初始值列表,多个值用逗号分隔。) |
| 74 | + |
| 75 | +在速记声明中,没有必要为数组中的每一个元素指定初始值。 |
| 76 | + |
| 77 | +```golang |
| 78 | +package main |
| 79 | + |
| 80 | +import ( |
| 81 | + "fmt" |
| 82 | +) |
| 83 | + |
| 84 | +func main() { |
| 85 | + a := [3]int{12} |
| 86 | + fmt.Println(a) |
| 87 | +} |
| 88 | +``` |
| 89 | + |
| 90 | +上面程序的第 8 行:`a := [3]int{12} `声明了一个长度为 3 的数组,但是只提供了一个初值 12。剩下的两个元素被自动赋值为 0。程序 的输出为:`[12 0 0]`。 |
| 91 | + |
| 92 | +在声明数组时你可以忽略数组的长度并用` ... `代替,让编译器为你自动推导数组的长度。比如下面的程序: |
| 93 | + |
| 94 | +```golang |
| 95 | +package main |
| 96 | + |
| 97 | +import ( |
| 98 | + "fmt" |
| 99 | +) |
| 100 | + |
| 101 | +func main() { |
| 102 | + a := [...]int{12, 78, 50} // ... makes the compiler determine the length |
| 103 | + fmt.Println(a) |
| 104 | +} |
| 105 | +``` |
| 106 | + |
| 107 | +上面已经提到,数组的长度是数组类型的一部分。因此 `[5]int` 和 `[25]int`是两个不同类型的数组。正是因为如此,一个数组不能动态改变长度。不要担心这个限制,因为切片(`slices`)可以弥补这个不足。 |
| 108 | + |
| 109 | +```golang |
| 110 | +package main |
| 111 | + |
| 112 | +func main() { |
| 113 | + a := [3]int{5, 78, 8} |
| 114 | + var b [5]int |
| 115 | + b = a //not possible since [3]int and [5]int are distinct types |
| 116 | +} |
| 117 | +``` |
| 118 | + |
| 119 | +在上面程序的第 6 行,我们试图将一个` [3]int `类型的数组赋值给一个 `[5]int `类型的数组,这是不允许的。编译会报错:`main.go:6: cannot use a (type [3]int) as type [5]int in assignment。` |
| 120 | + |
| 121 | +## 数组是值类型 |
| 122 | + |
| 123 | +在 Go 中数组是值类型而不是引用类型。这意味着当数组变量被赋值时,将会获得原数组(*译者注:也就是等号右面的数组*)的拷贝。新数组中元素的改变不会影响原数组中元素的值。 |
| 124 | + |
| 125 | +```golang |
| 126 | +package main |
| 127 | + |
| 128 | +import "fmt" |
| 129 | + |
| 130 | +func main() { |
| 131 | + a := [...]string{"USA", "China", "India", "Germany", "France"} |
| 132 | + b := a // a copy of a is assigned to b |
| 133 | + b[0] = "Singapore" |
| 134 | + fmt.Println("a is ", a) |
| 135 | + fmt.Println("b is ", b) |
| 136 | +} |
| 137 | +``` |
| 138 | + |
| 139 | +上面程序的第 7 行,将数组 a 的拷贝赋值给数组 b。第 8 行,b 的第一个元素被赋值为 `Singapore`。这将不会影响到原数组 a。程序的输出为: |
| 140 | + |
| 141 | +```golang |
| 142 | +a is [USA China India Germany France] |
| 143 | +b is [Singapore China India Germany France] |
| 144 | +``` |
| 145 | + |
| 146 | +同样的,如果将数组作为参数传递给函数,仍然是值传递,在函数中对(作为参数传入的)数组的修改不会造成原数组的改变。 |
| 147 | + |
| 148 | +```golang |
| 149 | +package main |
| 150 | + |
| 151 | +import "fmt" |
| 152 | + |
| 153 | +func changeLocal(num [5]int) { |
| 154 | + num[0] = 55 |
| 155 | + fmt.Println("inside function ", num) |
| 156 | + |
| 157 | +} |
| 158 | +func main() { |
| 159 | + num := [...]int{5, 6, 7, 8, 8} |
| 160 | + fmt.Println("before passing to function ", num) |
| 161 | + changeLocal(num) //num is passed by value |
| 162 | + fmt.Println("after passing to function ", num) |
| 163 | +} |
| 164 | +``` |
| 165 | + |
| 166 | +上面程序的第 13 行,数组 num 是通过值传递的方式传递给函数 `changeLocal` 的,因此该函数执行过程中不会造成 num 的改变。程序输出如下: |
| 167 | + |
| 168 | +```golang |
| 169 | +before passing to function [5 6 7 8 8] |
| 170 | +inside function [55 6 7 8 8] |
| 171 | +after passing to function [5 6 7 8 8] |
| 172 | +``` |
| 173 | + |
| 174 | +## 数组的长度 |
| 175 | + |
| 176 | +希望你喜欢阅读。请留下宝贵的意见和反馈:) |
0 commit comments