數組(array)
在GO語言中,數組是用于存儲相同數據類型的集合,數組長度必須是一個常量表達式,且是一個非負數
GO語言中的數組是一種值類型,下文會介紹
數組定義的格式
var 數組名稱 [數組長度]數組類型
例:
var arr [3]int
輸出為:
[0 0 0]
在GO語言中,初始化數組時,給定了數組長度,但沒給定數組下標的值,int
類型默認值為0
,string
類型默認為空值,這是常用的兩種類型
數組的賦值操作
// 第一種賦值操作
var arr = [3]int{1,2,3} //直接將值初始化,不同的類似初始化值不相同
// 第二種賦值操作
var arr [3]int // 初始化數組變量
arr[1] = 1 // 給數組的下標賦值
// 第三種賦值操作
var arr = [3]string{1:"one",2:"two"}
// 一般常用的就這三種,其它方式都是一些變相的操作,這里不做介紹
有必要說明一下:數組在初始中給定了數組的長度,用
len(arr)-1
可以得到數組的長度,如果賦值時操出數組的最大長度,在編譯的時候將會報錯
先來個練習加深下前面所說的影響,將一個int型數組的值全部乘以2
package main
import "fmt"
func main(){
var arrs = [5]int{1, 2, 3, 4, 5}
for i, v := range arrs {
arrs[i] = v * 2
fmt.Printf("index is %d,value is %d \n", i, arrs[i])
}
}
輸出為
index is 0,value is 2
index is 1,value is 4
index is 2,value is 6
index is 3,value is 8
index is 4,value is 10
數組的值類型
在GO語言中,數組的存儲是一種值的類型,不像C等其它語言是指向首元素的指針,所以在創建數組也可以通過new()
來創建一個指針數組
例:
var arr1 = new([5]int)
var arr2 [5]int
輸出分別為
&[0 0 0 0 0]
[0 0 0 0 0]
&
符號熟悉么,沒錯,他表明這是一個指針數組,在函數中傳遞時,不用將數組的值進行復制一遍
多維數組
在某種程度上面來說,多維數組其實就是將多個一維數組進行嵌套一種表達方式,例如:var arr1 [3][5]int
輸出為[[0 0 0 0 0] [0 0 0 0 0] [0 0 0 0 0]]
如果嘗過其它語言編程的人一看就明白是怎么回事了
做個小題目:定義一個空的二維數組,將二維數組的值與索引一至
package main
import "fmt"
func main{
var arr_more [5][5]int
fmt.Println(arr_more)
for i, x := range arr_more {
for i1, _ := range x {
arr_more[i][i1] = i1
}
}
fmt.Println(arr_more)
}
輸出
初始化數組:
[[0 0 0 0 0] [0 0 0 0 0] [0 0 0 0 0] [0 0 0 0 0] [0 0 0 0 0]]
值與索引對應
[[0 1 2 3 4] [0 1 2 3 4] [0 1 2 3 4] [0 1 2 3 4] [0 1 2 3 4]]
數組的處理還有很多別的形式,等著你在實際工作中的發現與總結
切片(slice)
切片是對數組一個連續片段的引用,它是一個引用類型,存儲的是指針,所以在性能上比值數組更快,使用方法和數組基本類似,也可以通過索引進行訪問,len()
獲得切片的長度,cap()
獲得切片最大的長度
切片聲明格式
var 聲明變量 []變量類型
和聲明數組非常相似,切片的聲明是不需要指定數組長度,因為切片的長度是可變的(等會介紹)
一個切片在初始化之前默認為nil
,長度為0
一個相對完整的例子:
package main
import "fmt"
func main{
var arr = [5]int{1, 2, 3, 4, 5}
var slice []int
fmt.Printf("初始化切片默認 %d\n", slice)
var slice1 []int = arr[:]
fmt.Printf("切片復制數組(簡寫) %d\n", slice1)
var slice2 []int = arr[0:2]
fmt.Printf("切片獲得數組0-1的下標 %d\n", slice2)
var slice3 []int = arr[2:5]
fmt.Printf("切片獲得數組2-4的下標 %d\n", slice3)
}
輸出
初始化切片默認 []
切片復制數組(簡寫) [1 2 3 4 5]
切片獲得數組0-1的下標 [1 2]
切片獲得數組2-4的下標 [3 4 5]
上面的例子介紹了切片的初始化和獲得數組中某一個片段的值,當然還有一些其他用法比如[:3]
,[3:]
等等,切片的靈活性非常高,就看你需要什么數組的哪個片段
注意 絕對不要用指針指向 slice。切片本身已經是一個引用類型,所以它本身就是一個指針!!
用make()創建一個切片
當相關的數組還沒有創建好的時候,我們可以用make()
函數來創建一個切片,同時創建好相關聯的數組
var slice []int = make([]type,len,cap)
var 切片變量 []切片類型 = make([]數組類型,數組長度,最大長度) cap是可選參數
通過一個例子來說明make的使用
package main
import "fmt"
func main() {
var slice1 []int = make([]int, 5)
for i := 0; i < len(slice1); i++ {
slice1[i] = 5 * i
fmt.Printf("Slice at %d is %d\n", i, slice1[i])
}
fmt.Printf("\nThe length of slice1 is %d\n", len(slice1))
fmt.Printf("The capacity of slice1 is %d\n", cap(slice1))
}
輸出
Slice at 0 is 0
Slice at 1 is 5
Slice at 2 is 10
Slice at 3 is 15
Slice at 4 is 20
The length of slice1 is 5
The capacity of slice1 is 5
make()函數在初始化切片是經常被使用的方式,當然你要用簡單的var slice[]int = arr[:]
也可以,看習慣
切片的復制與追加
通過copy()
與append()
來進行操作
一個例子:
package main
import "fmt"
func main(){
sl_from := []int{1, 2, 3}
sl_to := make([]int, 10)
n1 := copy(sl_to, sl_from)
fmt.Println(sl_to)
fmt.Printf("Copied %d elements\n", n1) // n == 3
sl3 := []int{1, 2, 3}
sl3 = append(sl3, 4, 5, 6)
fmt.Println(sl3)
}
輸出:
[1 2 3 0 0 0 0 0 0 0]
Copied 3 elements
[1 2 3 4 5 6]
復制與增加需要注意的:
- 是將后面的元素或切片追加到前面
- 必須是相同的元素類型
- 當容量不足時,會生成一個新的地址來保證新增加的元素
- 如果上面的條件都滿足,一般來說都會返回成功,除非內存耗盡了(無解)
小記:
當然切片還有更多的細節及處理方式,完整版最好還是去看官方提供的文檔,本篇重點介紹了GO的數組與切片,下一篇將介紹GO的map類型
如果覺得文章能夠對您有所幫助,可以關注我,你的支持會鼓勵我不斷分享更多更好的優質文章。