Array(数组)

数组是同一种数据类型元素的集合。 在Go语言中,数组从声明时就确定,使用时可以修改数组成员,但是数组大小不可变化。 基本语法:

  1. // 定义一个长度为3元素类型为int的数组a
  2. var a [3]int

数组定义:

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

比如:var a [5]int, 数组的长度必须是常量,并且长度是数组类型的一部分。一旦定义,长度不能变。 [5]int[10]int是不同的类型。

  1. var a [3]int
  2. var b [4]int
  3. a = b //不可以这样做,因为此时a和b是不同的类型

数组可以通过下标进行访问,下标是从0开始,最后一个元素下标是:len-1,访问越界(下标在合法范围之外),则触发访问越界,会panic

数组的初始化

数组的初始化也有很多方式。

方法一

初始化数组时可以使用初始化列表来设置数组元素的值。

  1. func main() {
  2. var testArray [3]int
  3. var numArray = [...]int{1, 2}
  4. var cityArray = [...]string{"北京", "上海", "深圳"}
  5. fmt.Println(testArray) //[0 0 0]
  6. fmt.Println(numArray) //[1 2]
  7. fmt.Printf("type of numArray:%T\n", numArray) //type of numArray:[2]int
  8. fmt.Println(cityArray) //[北京 上海 深圳]
  9. fmt.Printf("type of cityArray:%T\n", cityArray) //type of cityArray:[3]string
  10. }

方法二

按照上面的方法每次都要确保提供的初始值和数组长度一致,一般情况下我们可以让编译器根据初始值的个数自行推断数组的长度,例如:

  1. func main() {
  2. a := [...]int{1: 1, 3: 5}
  3. fmt.Println(a) // [0 1 0 5]
  4. fmt.Printf("type of a:%T\n", a) //type of a:[4]int
  5. }

方法三

我们还可以使用指定索引值的方式来初始化数组,例如:

  1. func main() {
  2. a := [...]int{1: 1, 3: 5}
  3. fmt.Println(a) // [0 1 0 5]
  4. fmt.Printf("type of a:%T\n", a) //type of a:[4]int
  5. }

数组的遍历

  1. func main() {
  2. var a = [...]string{"北京", "上海", "深圳"}
  3. // 方法1:for循环遍历
  4. for i := 0; i < len(a); i++ {
  5. fmt.Println(a[i])
  6. }
  7. // 方法2:for range遍历
  8. for index, value := range a {
  9. fmt.Println(index, value)
  10. }
  11. }

多维数组

Go语言是支持多维数组的,我们这里以二维数组为例(数组中又嵌套数组)。

二维数组的定义

  1. func main() {
  2. a := [3][2]string{
  3. {"北京", "上海"},
  4. {"广州", "深圳"},
  5. {"成都", "重庆"},
  6. }
  7. fmt.Println(a) //[[北京 上海] [广州 深圳] [成都 重庆]]
  8. fmt.Println(a[2][1]) //支持索引取值:重庆
  9. }

二维数组的遍历

  1. func main() {
  2. a := [3][2]string{
  3. {"北京", "上海"},
  4. {"广州", "深圳"},
  5. {"成都", "重庆"},
  6. }
  7. for _, v1 := range a {
  8. for _, v2 := range v1 {
  9. fmt.Printf("%s\t", v2)
  10. }
  11. fmt.Println()
  12. }
  13. }
  1. 输出:
  2. 北京 上海 广州 深圳 成都 重庆

注意: 多维数组只有第一层可以使用…来让编译器推导数组长度。例如:

  1. //支持的写法
  2. a := [...][2]string{
  3. {"北京", "上海"},
  4. {"广州", "深圳"},
  5. {"成都", "重庆"},
  6. }
  7. //不支持多维数组的内层使用...
  8. b := [3][...]string{
  9. {"北京", "上海"},
  10. {"广州", "深圳"},
  11. {"成都", "重庆"},
  12. }

数组是值类型

数组是值类型,赋值和传参会复制整个数组。因此改变副本的值,不会改变本身的值。

  1. func modifyArray(x [3]int) {
  2. x[0] = 100
  3. }
  4. func modifyArray2(x [3][2]int) {
  5. x[2][0] = 100
  6. }
  7. func main() {
  8. a := [3]int{10, 20, 30}
  9. modifyArray(a) //在modify中修改的是a的副本x
  10. fmt.Println(a) //[10 20 30]
  11. b := [3][2]int{
  12. {1, 1},
  13. {1, 1},
  14. {1, 1},
  15. }
  16. modifyArray2(b) //在modify中修改的是b的副本x
  17. fmt.Println(b) //[[1 1] [1 1] [1 1]]
  18. }

注意:

  1. 数组支持 “==“、”!=” 操作符,因为内存总是被初始化过的。
  2. [n]T表示指针数组,[n]T表示数组指针 。

    练习题

  3. 求数组[1, 3, 5, 7, 8]所有元素的和

  4. 找出数组中和为指定值的两个元素的下标,比如从数组[1, 3, 5, 7, 8]中找出和为8的两个元素的下标分别为(0,3)和(1,2)。

  1. var array1 = [...]int{1, 3, 5, 7, 8}
  2. func arraysum() {
  3. sum := 0
  4. for _, array2 := range array1 {
  5. sum = sum + array2
  6. }
  7. fmt.Print(sum)
  8. }
  9. func arrayflag() {
  10. for i := 0; i < len(array1); i++ {
  11. for m := i + 1; m < len(array1); m++ {
  12. if array1[i]+array1[m] == 8 {
  13. fmt.Printf("(%d,%d)", i, m)
  14. }
  15. }
  16. }
  17. }