數組與切片
package main
import "fmt"
func main() {
var a [2]string
a[0] = "Hello"
a[1] = "World"
fmt.Println(a[0], a[1])
fmt.Println(a)
primes := [6]int{2, 3, 5, 7, 11, 13}
x := primes[1:4]
y := primes[2:5]
y[0] = 1000
fmt.Println(primes)
fmt.Println(x)
fmt.Println(y)
}
//輸出
Hello World
[Hello World]
[2 3 1000 7 11 13]
[3 1000 7]
[1000 7 11]
切片操作很像python,左開右閉。但是切片還和原數組保持公用數據,相當於這段地址的別名。
s := []struct {
i int
b bool
}{
{2, true},
{3, false},
{5, true},
{7, true},
{11, false},
{13, true},
}
go可以給數組用一種非常奇怪的初始化
s := []int{2, 3, 5, 7, 11, 13}
s = s[1:4] // [3 5 7]
s = s[:2] // [3 5]
此時數組s變了,也可以像python一樣有缺省
長度與容量
package main
import "fmt"
func main() {
s := []int{2, 3, 5, 7, 11, 13}
printSlice(s)
// Slice the slice to give it zero length.
s = s[:0]
printSlice(s)
// Extend its length.
s = s[:4]
printSlice(s)
// Drop its first two values.
s = s[2:]
printSlice(s)
}
func printSlice(s []int) {
fmt.Printf("len=%d cap=%d %v\n", len(s), cap(s), s)
}
//輸出
len=6 cap=6 [2 3 5 7 11 13]
len=0 cap=6 []
len=4 cap=6 [2 3 5 7]
len=2 cap=4 [5 7]
可以看到,只有在切掉前面的時候容量減少,長度減少不影響原來存在的值。
切片和數組中length只決定能訪問到的長度,而容量則表示內存中有的真實情況。
構建切片
package main
import "fmt"
func main() {
a := make([]int, 5)
printSlice("a", a)
b := make([]int, 0, 5)
printSlice("b", b)
c := b[:2]
printSlice("c", c)
d := c[2:5]
printSlice("d", d)
}
func printSlice(s string, x []int) {
fmt.Printf("%s len=%d cap=%d %v\n",
s, len(x), cap(x), x)
}
// 輸出
a len=5 cap=5 [0 0 0 0 0]
b len=0 cap=5 []
c len=2 cap=5 [0 0]
d len=3 cap=3 [0 0 0]
用make函數構建可以指定len和cap關鍵字。
切片的數據類型
切片可以包含任何數據類型
[外鏈圖片轉存失敗,源站可能有防盜鏈機制,建議將圖片保存下來直接上傳(img-5ZlAwwdL-1588406777617)(./…/img/slice%20of%20slice.jpg)]
嚇得我以爲可以閉包,但是顯然沒有。
官方的意思:
board := [][]string{
[]string{"_", "_", "_"},
[]string{"_", "_", "_"},
[]string{"_", "_", "_"},
}
我的理解是像lisp那樣car和cdr是不同值:
(cons
[1 2 3 4]
[1.0 2.0 3.0 4.0])
遍歷
package main
import "fmt"
var pow = []int{1, 2, 4, 8, 16, 32, 64, 128}
func main() {
for i, v := range pow {
fmt.Printf("2**%d = %d\n", i, v)
}
}
// 輸出
2**0 = 1
2**1 = 2
2**2 = 4
2**3 = 8
2**4 = 16
2**5 = 32
2**6 = 64
2**7 = 128
這裏,range關鍵字對切片操作的返回值爲其下標和對應的值,加上我們前面學習的_
關鍵字可以當作不用那麼也可以寫成:
for _, v := range pow {
fmt.Println(v)
}
for i, _ := range pow {
fmt.Println(i)
}
for i := range pow {
fmt.Println(i)
}