加入收藏 | 设为首页 | 会员中心 | 我要投稿 PHP编程网 - 黄冈站长网 (http://www.0713zz.com/)- 数据应用、建站、人体识别、智能机器人、语音技术!
当前位置: 首页 > 综合聚焦 > 编程要点 > 语言 > 正文

Go 中的可寻址和不可寻址如何理解?

发布时间:2021-11-06 04:15:42 所属栏目:语言 来源:互联网
导读:1. 什么叫可寻址? 可直接使用 操作符取地址的对象,就是可寻址的(Addressable)。比如下面这个例子 func main() { name := iswbm fmt.Println(name) // output: 0xc000010200 } 程序运行不会报错,说明 name 这个变量是可寻址的。 但不能说 iswbm 这个字符串
 1. 什么叫可寻址?
可直接使用 & 操作符取地址的对象,就是可寻址的(Addressable)。比如下面这个例子
 
func main() {
    name := "iswbm"
    fmt.Println(&name)  
    // output: 0xc000010200
}
程序运行不会报错,说明 name 这个变量是可寻址的。
 
但不能说 "iswbm" 这个字符串是可寻址的。
 
"iswbm" 是字符串,字符串都是不可变的,是不可寻址的,后面会介绍到。
 
在开始逐个介绍之前,先说一下结论
 
指针可以寻址:&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
}
数组元素索引: &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()
}
注意上面写法与这个写法的区别,下面这个写法代表不同意思,其中的 & 并不是取地址的操作,而代表实例化一个结构体的指针。
 
type Profile struct {
    Name string
}
 
func main() {
    fmt.Println(&Profile{Name: "iswbm"}) // ok
}
虽然组合字面量是不可寻址的,但却可以对组合字面量的字段属性进行寻址(直接访问)
 
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()
}
函数或方法
 
func getStr() string {
    return "iswbm"
}
func main() {
    fmt.Println(&getStr)
    // cannot take the address of getStr
}
基本类型字面量
 
字面量分:基本类型字面量 和 复合型字面量。
 
基本类型字面量,是一个值的文本表示,都是不应该也是不可以被寻址的。
 
func getInt() int {
    return 1024
}
 
func main() {
    fmt.Println(&getInt())
    // cannot take the address of getInt()
}
map 中的元素
 
字典比较特殊,可以从两个角度来反向推导,假设字典的元素是可寻址的,会出现 什么问题?
 
如果字典的元素不存在,则返回零值,而零值是不可变对象,如果能寻址问题就大了。
 
而如果字典的元素存在,考虑到 Go 中 map 实现中元素的地址是变化的,这意味着寻址的结果也是无意义的。
 
基于这两点,Map 中的元素不可寻址,符合常理。
 
func main() {
    p := map[string]string {
        "name": "iswbm",
    }
 
    fmt.Println(&p["name"])
    // cannot take the address of p["name"]
}
搞懂了这点,你应该能够理解下面这段代码为什么会报错啦~
 
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"},
    }
 
    //编译错误:cannot assign to struct field m[1].Name in map
    m[1].Name = "Scrapup"
数组字面量
 
数组字面量是不可寻址的,当你对数组字面量进行切片操作,其实就是寻找内部元素的地址,下面这段代码是会报错的
 
func main() {
    fmt.Println([3]int{1, 2, 3}[1:])
    // invalid operation [3]int literal[1:] (slice of unaddressable value)
}
是不是很简单?跟着明哥一起来攻克 Go 的各个边边角角的知识吧!

(编辑:PHP编程网 - 黄冈站长网)

【声明】本站内容均来自网络,其相关言论仅代表作者个人观点,不代表本站立场。若无意侵犯到您的权利,请及时与联系站长删除相关内容!

    热点阅读