解码接口运行类型断言的编译器输出
来源:stackoverflow
时间:2024-03-31 09:00:31 417浏览 收藏
偷偷努力,悄无声息地变强,然后惊艳所有人!哈哈,小伙伴们又来学习啦~今天我将给大家介绍《解码接口运行类型断言的编译器输出》,这篇文章主要会讲到等等知识点,不知道大家对其都有多少了解,下面我们就一起来看一吧!当然,非常希望大家能多多评论,给出合理的建议,我们一起学习,一起进步!
我最近在使用 Atomic.Value
的 Load()
方法时遇到空接口。我正在尝试空接口类型断言 - https://play.golang.org/p/CLyY2y9-2VF
这激起了我的兴趣,我决定在幕后看看编译器会采取哪些操作,以便代码在尝试读取 nil
接口 {} 上的具体值时不会出现恐慌(例如,当 Store 尚未被调用时,您调用 Load.(type)
时)。
我可以看到,在不安全版本中,编译器有这个汇编指令导致恐慌:调用runtime.panicdottypeE(SB)
安全版本中显然不存在紧急指令。有人可以更详细地解释一下当我们使用 ok 捕获返回值时编译器正在做什么吗(或许还可以向我指出 godbolt 链接中相应的汇编指令)?
以下是不安全版本 [1] 和安全版本 [2] 的 godbolt 编译器链接。
[1] https://godbolt.org/z/76onvj
[2] https://godbolt.org/z/e8aoqe
解决方案
空接口类型(在运行时包中称为 eface
)是 2 个指针,第一个指向基础类型(例如 bool 类型、int 类型、yourstruct 类型...),第二个是指向数据的指针(iirc 在某些情况下数据本身)。
不安全版本:
call "".returnemptyinterface(sb) call the function pcdata $0, $1 pcdata is not a real instruction, ignore movq 8(sp), ax ax <- pointer to data movq (sp), cx cx <- pointer to type pcdata $0, $2 leaq type.bool(sb), dx dx <- pointer to bool type cmpq cx, dx compare cx and dx jne main_pc156 if they are not equal jump to main_pc156
在main_pc156中,编译器将调用runtime.panicdottypee,这基本上会出现恐慌。源代码来自runtime/iface.go
(可以在你的$gopath/src
中找到它):
func panicdottypee(have, want, iface *_type) { panic(&typeassertionerror{iface, have, want, ""}) }
安全版本:
pcdata $0, $0 pcdata $1, $0 call "".returnemptyinterface(sb) call the function pcdata $0, $1 movq 8(sp), ax ax <- pointer to data movq (sp), cx cx <- pointer to type main_pc47: pcdata $0, $2 leaq type.bool(sb), dx dx <- pointer to bool type cmpq cx, dx compare cx and dx jne main_pc186 if not equal jump main_pc186 pcdata $0, $3 movblzx (ax), ax ax <- dereference ax main_pc62:
和main_pc186
pcdata $0, $3 xorl ax, ax ax <- 0 jmp main_pc62 jump back at the end of previous block
这里ax
对应于代码中的x
,但是什么对应于代码中的ok
呢?没有什么!如果您检查 println
的代码,您会看到:
cmpq cx, dx compare cx and dx seteq al al <- 1 if cx equal to dx otherwise 0
因此编译器决定在打印时再次比较它们。
快速总结:它们执行完全相同的操作,只是 ok == false
的情况不同。
这两个位有以下共同点:
pcdata $0, $0 pcdata $1, $0 call "".returnemptyinterface(sb) pcdata $0, $1 movq 8(sp), ax movq (sp), cx pcdata $0, $2 leaq type.bool(sb), dx cmpq cx, dx jne main_pc156 <==== jump pcdata $0, $3 movblzx (ax), ax
在main_pc156
中找到的代码是我们关心的部分。正如您所注意到的,对于单值类型断言,这是:
main_pc156: movq dx, 8(sp) pcdata $0, $1 leaq type.interface {}(sb), ax pcdata $0, $0 movq ax, 16(sp) call runtime.panicdottypee(sb) xchgl ax, ax
这是不可避免的,一旦我们跳转到 main_pc156
,我们就会恐慌。
另一方面,二值类型断言的代码是:
main_pc186: pcdata $0, $3 xorl AX, AX jmp main_pc62
这与之前的情况有很大不同,它让我们回到第一段代码的末尾,恢复执行。
理论要掌握,实操不能落!以上关于《解码接口运行类型断言的编译器输出》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!
-
502 收藏
-
502 收藏
-
501 收藏
-
501 收藏
-
501 收藏
-
139 收藏
-
204 收藏
-
325 收藏
-
477 收藏
-
486 收藏
-
439 收藏
-
357 收藏
-
352 收藏
-
101 收藏
-
440 收藏
-
212 收藏
-
143 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 508次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习