一文理解Go 中的可寻址和不可寻址
来源:脚本之家
时间:2023-01-07 12:14:09 205浏览 收藏
对于一个Golang开发者来说,牢固扎实的基础是十分重要的,golang学习网就来带大家一点点的掌握基础知识点。今天本篇文章带大家了解《一文理解Go 中的可寻址和不可寻址》,主要介绍了可寻址、不可寻址,希望对大家的知识积累有所帮助,快点收藏起来吧,否则需要时就找不到了!
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 中的可寻址和不可寻址》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!
-
505 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
416 收藏
-
189 收藏
-
213 收藏
-
315 收藏
-
381 收藏
-
128 收藏
-
236 收藏
-
340 收藏
-
298 收藏
-
249 收藏
-
460 收藏
-
495 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习