理解Golang中的數(shù)組(array)、切片(slice)和map
我比較喜歡先給出代碼,然后得出結(jié)論
數(shù)組
package main
import (
"fmt"
)
func main() {
arr := [...]int{1, 2, 3}
//打印初始的指針
fmt.Printf("the pointer is : %p \n", &arr)
printPointer(arr)
}
func printPointer(any interface{}) {
fmt.Printf("the pointer is : %p \n", &any)
}
結(jié)果
1 the pointer is : 0xc082008580
2 the pointer is : 0xc0820001d0
切片
package main
import (
"fmt"
)
func main() {
arr := make([]int, 3)
//打印初始的指針
fmt.Printf("the pointer is : %p \n", arr)
printPointer(arr)
}
func printPointer(any interface{}) {
fmt.Printf("the pointer is : %p \n", any)
}
結(jié)果
1 the pointer is : 0xc082008580
2 the pointer is : 0xc082008580
map
package main
import (
"fmt"
)
func main() {
arr := make(map[int]string)
//arr := [3]int{1, 2, 3}
//打印初始的指針
fmt.Printf("the pointer is : %p \n", arr)
printPointer(arr)
}
func printPointer(any interface{}) {
fmt.Printf("the pointer is : %p \n", any)
}
運(yùn)行結(jié)果
1 the pointer is : 0xc082007c80
2 the pointer is : 0xc082007c80
由此,我們看到數(shù)組本身傳過(guò)去的是值,傳到函數(shù)之后,被開(kāi)辟了另外一個(gè)空間。
因?yàn)閿?shù)組就是他本身。這一句好像不太好理解。
這是切片 arr := make([]int, 3) 而arr 本身不是一個(gè)數(shù)組,至少不是我們所想要的指向的一個(gè)數(shù)組。只是arr里有一個(gè)地址指向數(shù)組。
這么舉個(gè)例子:
arr := [...]int{1,2,3,4,5} 這是一個(gè)數(shù)組,懂得go語(yǔ)言的都明白。 arr本身就是數(shù)組
arrSlice := arr[0:5] 這是一個(gè)切片。 打印所得的值是一樣的,和上面。 arrSlice本身不是數(shù)組,只是arrSlice本身有一個(gè)值是指向arr的指針。
切片是指一個(gè)結(jié)構(gòu)體,大體結(jié)構(gòu)像這樣:
struct slice{
ptr *Elem
len int
cap int
}
也就是說(shuō),上面的arrSlice其實(shí)是一個(gè)結(jié)構(gòu)體。里面有一個(gè)屬性 ptr指向數(shù)組 arr
其實(shí)arrSlice也是傳到函數(shù)里,也是進(jìn)行了復(fù)制。但是盡管傳過(guò)去是一個(gè)復(fù)制的結(jié)構(gòu)體,他的屬性ptr,沒(méi)有變。還是一個(gè)指向原數(shù)組的指針。
下面的例子見(jiàn)證他自己傳過(guò)去,是一個(gè)復(fù)制的過(guò)程:
package main
import (
"fmt"
)
func main() {
arrSlice := make([]int, 4)
fmt.Printf("the pointer is : %p \n", arrSlice)
fmt.Printf("the pointer is : %p \n", &arrSlice) //這是arrSlice本身的指針,也就是結(jié)構(gòu)體的指針
printPointer(arrSlice)
}
func printPointer(any interface{}) {
fmt.Printf("the pointer is : %p \n", any)
fmt.Printf("the pointer is : %p \n", &any) //打印傳過(guò)來(lái)的結(jié)構(gòu)體arrSlice的指針
}
看結(jié)果:
the pointer is : 0xc0820085a0
the pointer is : 0xc082008580
the pointer is : 0xc0820085a0
the pointer is : 0xc0820001d0
第1、3個(gè)的打印是打印這個(gè)結(jié)構(gòu)體的ptr屬性,也就是指向數(shù)組的指針。
其實(shí)這個(gè)結(jié)構(gòu)體傳到函數(shù)里,是一個(gè)復(fù)制的過(guò)程,第2、4的指針不一樣。
大家在對(duì)照下面的圖片理解一下:
相關(guān)文章
詳解Go并發(fā)編程時(shí)如何避免發(fā)生競(jìng)態(tài)條件和數(shù)據(jù)競(jìng)爭(zhēng)
大家都知道,Go是一種支持并發(fā)編程的編程語(yǔ)言,但并發(fā)編程也是比較復(fù)雜和容易出錯(cuò)的。比如本篇分享的問(wèn)題:競(jìng)態(tài)條件和數(shù)據(jù)競(jìng)爭(zhēng)的問(wèn)題2023-04-04Go標(biāo)準(zhǔn)庫(kù)encoding/gob的具體使用
Go標(biāo)準(zhǔn)庫(kù)encoding/gob實(shí)現(xiàn)二進(jìn)制序列化與反序列化,本文主要介紹了Go標(biāo)準(zhǔn)庫(kù)encoding/gob的具體使用,感興趣的可以了解一下2025-06-06Go語(yǔ)言實(shí)現(xiàn)對(duì)XML的讀取和修改
這篇文章主要為大家詳細(xì)介紹了Go語(yǔ)言實(shí)現(xiàn)對(duì)XML的讀取和修改的相關(guān)知識(shí),文中的示例代碼講解詳細(xì),感興趣的小伙伴可以跟隨小編一起學(xué)習(xí)一下2023-12-12Go語(yǔ)言使用kafka-go實(shí)現(xiàn)Kafka消費(fèi)消息
本篇文章主要介紹了使用kafka-go庫(kù)消費(fèi)Kafka消息,包含F(xiàn)etchMessage和ReadMessage的區(qū)別和適用場(chǎng)景,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的可以了解一下2024-12-12模塊一 GO語(yǔ)言基礎(chǔ)知識(shí)-庫(kù)源碼文件
這篇文章主要介紹了模塊一 GO語(yǔ)言基礎(chǔ)知識(shí)-庫(kù)源碼文件,本文給大家介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下2021-01-01在ubuntu下安裝go開(kāi)發(fā)環(huán)境的全過(guò)程
Go語(yǔ)言是谷歌公司開(kāi)發(fā)的編程語(yǔ)言,雖然安裝和配置go很簡(jiǎn)單,但是很多初學(xué)者在第一次安裝go環(huán)境時(shí)會(huì)遇到各種坑,下面這篇文章主要給大家介紹了關(guān)于在ubuntu下安裝go開(kāi)發(fā)環(huán)境的相關(guān)資料,文中通過(guò)圖文介紹的非常詳細(xì),需要的朋友可以參考下2022-08-08golang 實(shí)現(xiàn)json類型不確定時(shí)的轉(zhuǎn)換
這篇文章主要介紹了golang 實(shí)現(xiàn)json類型不確定時(shí)的轉(zhuǎn)換操作,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過(guò)來(lái)看看吧2021-01-01