文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

一文理解Go 中的可寻址和不可寻址

2024-04-02 19:55

关注

1. 什么叫可寻址?

可直接使用 & 操作符取地址的对象,就是可寻址的(Addressable)。比如下面这个例子


func main() { 
    name := "iswbm" 
    fmt.Println(&name)  
    // output: 0xc000010200 
} 

程序运行不会报错,说明 name 这个变量是可寻址的。

但不能说 "iswbm" 这个字符串是可寻址的。

"iswbm" 是字符串,字符串都是不可变的,是不可寻址的,后面会介绍到。

在开始逐个介绍之前,先说一下结论

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 中的可寻址和不可寻址的文章就介绍到这了,更多相关Go 中可寻址和不可寻址内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     807人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     351人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     314人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     433人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     221人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯