使用append向slice追加元素遇到的坑,在此總結(jié)一下。
一,細(xì)節(jié)1
urls := make(map[string]string, 3)
urls["baidu"] = "www.baidu.com"
urls["google"] = "www.google.com"
urls["csdn"] = "www.csdn.net"
names := make([]string, len(urls))
for key, _ := range urls {
names = append(names, key)
}
fmt.Println(names,len(names))
輸出結(jié)果
[ csdn baidu google] 6
前面多了幾個(gè)空格,長度為6,與預(yù)期的結(jié)果不一致
修改代碼
names := make([]string, 0)
for key, _ := range urls {
names = append(names, key)
}
fmt.Println(names,len(names))
或者
var names []string
for key, _ := range urls {
names = append(names, key)
}
fmt.Println(names,len(names))
輸出結(jié)果
[baidu google csdn] 3
總結(jié):append無論如何都是從slice的尾部開始追加數(shù)據(jù),原來的slice只有3個(gè)長度,現(xiàn)在每一次append都要重新分配一次內(nèi)存
二、細(xì)節(jié)2
s1 := []int{1, 2, 3}
s2 := []int{4, 5}
s1 = append(s1, s2)
fmt.Println(s1)
程序報(bào)錯(cuò):cannot use s2 (type []int) as type int in append
修改為:
s1 := []int{1, 2, 3}
s2 := []int{4, 5}
s1 = append(s1, s2...)
fmt.Println(s1)
輸出結(jié)果:
[1 2 3 4 5]
總結(jié):append切片時(shí)候別漏了'...'
三、細(xì)節(jié)3
new與make區(qū)別:
new只分配內(nèi)存它并不初始化內(nèi)存,只是將其置零。new(T)會(huì)為T類型的新項(xiàng)目,分配被置零的存儲(chǔ),并且返回它的地址,一個(gè)類型為T的值,也即其返回一個(gè)指向新分配的類型為T的指針,這個(gè)指針指向的內(nèi)容的值為零(zero value),注意并不是指針為零。比如,對于bool類型,零值為false;int的零值為0;string的零值是空字符串。
make用于slice,map,和channel的初始化,返回一個(gè)初始化的(而不是置零),類型為T的值(而不是T)。之所以有所不同,是因?yàn)檫@三個(gè)類型是使用前必須初始化的數(shù)據(jù)結(jié)構(gòu)。例如,slice是一個(gè)三元描述符,包含一個(gè)指向數(shù)據(jù)(在數(shù)組中)的指針,長度,以及容量,在這些項(xiàng)被初始化之前,slice都是nil的。對于slice,map和channel,make初始化這些內(nèi)部數(shù)據(jù)結(jié)構(gòu),并準(zhǔn)備好可用的值。
p := new([]int) //p == nil; with len and cap 0,被置零的slice結(jié)構(gòu)體的指針,即指向值為nil的slice的指針
fmt.Println(p)
v := make([]int, 10, 50) // v is initialed with len 10, cap 50
fmt.Println(v)
輸出結(jié)果:
&[]
[0 0 0 0 0 0 0 0 0 0]
new1 := new([2]int)
fmt.Println(new1)
new1[0] = 1
new1[1] = 2
fmt.Println(new1)
輸出結(jié)果:
&[0 0]
&[1 2]