一文理解Go 中的可尋址和不可尋址
1. 什么叫可尋址?
可直接使用 & 操作符取地址的對(duì)象,就是可尋址的(Addressable)。比如下面這個(gè)例子
func main() {
name := "iswbm"
fmt.Println(&name)
// output: 0xc000010200
}
程序運(yùn)行不會(huì)報(bào)錯(cuò),說(shuō)明 name 這個(gè)變量是可尋址的。
但不能說(shuō) "iswbm" 這個(gè)字符串是可尋址的。
"iswbm" 是字符串,字符串都是不可變的,是不可尋址的,后面會(huì)介紹到。
在開(kāi)始逐個(gè)介紹之前,先說(shuō)一下結(jié)論
- 指針可以尋址:&Profile{}
- 變量可以尋址:name := Profile{}
- 字面量通通不能尋址:Profile{}
2. 哪些是可以尋址的?
變量:&x
func main() {
name := "iswbm"
fmt.Println(&name)
// output: 0xc000010200
}
指針:&*x
type Profile struct {
Name string
}
func main() {
fmt.Println(unsafe.Pointer(&Profile{Name: "iswbm"}))
// output: 0xc000108040
}
數(shù)組元素索引: &a[0]
func main() {
s := [...]int{1,2,3}
fmt.Println(&s[0])
// output: xc0000b4010
}
切片
func main() {
fmt.Println([]int{1, 2, 3}[1:])
}
切片元素索引:&s[1]
func main() {
s := make([]int , 2, 2)
fmt.Println(&s[0])
// output: xc0000b4010
}
組合字面量: &struct{X type}{value}
所有的組合字面量都是不可尋址的,就像下面這樣子
type Profile struct {
Name string
}
func new() Profile {
return Profile{Name: "iswbm"}
}
func main() {
fmt.Println(&new())
// cannot take the address of new()
}
注意上面寫法與這個(gè)寫法的區(qū)別,下面這個(gè)寫法代表不同意思,其中的 & 并不是取地址的操作,而代表實(shí)例化一個(gè)結(jié)構(gòu)體的指針。
type Profile struct {
Name string
}
func main() {
fmt.Println(&Profile{Name: "iswbm"}) // ok
}
雖然組合字面量是不可尋址的,但卻可以對(duì)組合字面量的字段屬性進(jìn)行尋址(直接訪問(wèn))
type Profile struct {
Name string
}
func new() Profile {
return Profile{Name: "iswbm"}
}
func main() {
fmt.Println(new().Name)
}
3. 哪些是不可以尋址的?
常量
import "fmt"
const VERSION = "1.0"
func main() {
fmt.Println(&VERSION)
}
字符串
func getStr() string {
return "iswbm"
}
func main() {
fmt.Println(&getStr())
// cannot take the address of getStr()
}
函數(shù)或方法
func getStr() string {
return "iswbm"
}
func main() {
fmt.Println(&getStr)
// cannot take the address of getStr
}
基本類型字面量
字面量分:基本類型字面量 和 復(fù)合型字面量。
基本類型字面量,是一個(gè)值的文本表示,都是不應(yīng)該也是不可以被尋址的。
func getInt() int {
return 1024
}
func main() {
fmt.Println(&getInt())
// cannot take the address of getInt()
}
map 中的元素
字典比較特殊,可以從兩個(gè)角度來(lái)反向推導(dǎo),假設(shè)字典的元素是可尋址的,會(huì)出現(xiàn) 什么問(wèn)題?
如果字典的元素不存在,則返回零值,而零值是不可變對(duì)象,如果能尋址問(wèn)題就大了。
而如果字典的元素存在,考慮到 Go 中 map 實(shí)現(xiàn)中元素的地址是變化的,這意味著尋址的結(jié)果也是無(wú)意義的。
基于這兩點(diǎn),Map 中的元素不可尋址,符合常理。
func main() {
p := map[string]string {
"name": "iswbm",
}
fmt.Println(&p["name"])
// cannot take the address of p["name"]
}
搞懂了這點(diǎn),你應(yīng)該能夠理解下面這段代碼為什么會(huì)報(bào)錯(cuò)啦~
package main
import "fmt"
type Person struct {
Name string
Email string
}
func main() {
m := map[int]Person{
1:Person{"Andy", "1137291867@qq.com"},
2:Person{"Tiny", "qishuai231@gmail.com"},
3:Person{"Jack", "qs_edu2009@163.com"},
}
//編譯錯(cuò)誤:cannot assign to struct field m[1].Name in map
m[1].Name = "Scrapup"
數(shù)組字面量
數(shù)組字面量是不可尋址的,當(dāng)你對(duì)數(shù)組字面量進(jìn)行切片操作,其實(shí)就是尋找內(nèi)部元素的地址,下面這段代碼是會(huì)報(bào)錯(cuò)的
func main() {
fmt.Println([3]int{1, 2, 3}[1:])
// invalid operation [3]int literal[1:] (slice of unaddressable value)
}
到此這篇關(guān)于一文理解Go 中的可尋址和不可尋址的文章就介紹到這了,更多相關(guān)Go 中可尋址和不可尋址內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
golang jwt+token驗(yàn)證的實(shí)現(xiàn)
這篇文章主要介紹了golang jwt+token驗(yàn)證的實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-10-10
vscode配置go開(kāi)發(fā)環(huán)境的實(shí)戰(zhàn)過(guò)程
vscode配置go的開(kāi)發(fā)環(huán)境很簡(jiǎn)單,下面這篇文章主要給大家介紹了關(guān)于vscode配置go開(kāi)發(fā)環(huán)境的實(shí)戰(zhàn)過(guò)程,文中通過(guò)實(shí)例代碼介紹的非常詳細(xì),需要的朋友可以參考下2022-06-06
Go?web中cookie值安全securecookie庫(kù)使用原理
這篇文章主要為大家介紹了Go?web中cookie值安全securecookie庫(kù)使用及實(shí)現(xiàn)原理詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-11-11

