在 go 语言中 defer 是一个非常有意思的关键字特性。例子如下: 输出结果是: 在前几天我的读者群内有小伙伴讨论起了下面这个问题: 简单来讲,问题就是针对在 因为在 Go 语言的底层数据结构设计上 defer 是链表的数据结构: 大家担心如果循环过大 defer 链表会巨长,不够 “精益求精”。又或是猜想会不会 Go defer 的设计和 Redis 数据结构设计类似,自己做了优化,其实没啥大影响? 今天这篇文章,我们就来探索循环 Go defer,造成底层链表过长会不会带来什么问题,若有,具体有什么影响? 开始吸鱼之路。 在早年 Go1.13 时曾经对 defer 进行了一轮性能优化,在大部分场景下 提高了 defer 30% 的性能: 我们来回顾一下 Go1.13 的变更,看看 Go defer 优化在了哪里,这是问题的关键点。 在 Go1.12 及以前,调用 Go defer 时汇编代码如下: 在 Go1.13 及以后,调用 Go defer 时汇编代码如下: 从汇编的角度来看,像是原本调用 我们抱着疑问继续看下去。 相较于以前的版本,Go defer 的最小单元 该字段用于标识这个 这一块代码挺常规的,主要是获取调用 这个 可以看到它把 问题来了,它又在哪里处理分配到堆上的应用场景呢? 具体的 非常明确,先前的版本中调用的 主要优化在于其 defer 对象的堆栈分配规则的改变,措施是: 如果 Go 编译器检测到循环深度(loopdepth)为 1,则设置逃逸分析的结果,将分配到栈上,否则分配到堆上。 以此免去了以前频繁调用 回到问题本身,知道了 defer 优化的原理后。那 “循环里搞 defer 关键字,是否会造成什么性能影响?” 最直接的影响就是这大约 30% 的性能优化直接全无,且由于姿势不正确,理论上 defer 既有的开销(链表变长)也变大,性能变差。 因此我们要避免以下两种场景的代码: 第一个例子是直接在代码的 这个也是最常见的模式,无论是写爬虫时,又或是 Goroutine 调用时,不少人都喜欢这么写。 这属于显式的调用了循环。 第二个例子是在代码中使用类似 这种写法比较少见,因为 这属于隐式的调用,造成了类循环的作用。 显然,Defer 在设计上并没有说做的特别的奇妙。他主要是根据实际的一些应用场景进行了优化,达到了较好的性能。 虽然本身 defer 会带一点点开销,但并没有想象中那么的不堪使用。除非你 defer 所在的代码是需要频繁执行的代码,才需要考虑去做优化。 否则没有必要过度纠结,在实际上,猜测或遇到性能问题时,看看 PProf 的分析,看看 defer 是不是在相应的 hot path 之中,再进行合理优化就好。 所谓的优化,可能也只是去掉 defer 而采用手动执行,并不复杂。在编码时避免踩到 defer 的显式和隐式循环这 2 个雷区就可以达到性能最大化了。 更多golang相关技术文章,请访问golang教程栏目!package main
import "fmt"
func main() {
defer fmt.Println("煎鱼了")
fmt.Println("脑子进")
}脑子进
煎鱼了

for 循环里搞 defer 关键字,是否会造成什么性能影响?
defer 性能优化 30%

以前和现在对比
0x0070 00112 (main.go:6) CALL runtime.deferproc(SB)
0x0075 00117 (main.go:6) TESTL AX, AX
0x0077 00119 (main.go:6) JNE 137
0x0079 00121 (main.go:7) XCHGL AX, AX
0x007a 00122 (main.go:7) CALL runtime.deferreturn(SB)
0x007f 00127 (main.go:7) MOVQ 56(SP), BP 0x006e 00110 (main.go:4) MOVQ AX, (SP)
0x0072 00114 (main.go:4) CALL runtime.deferprocStack(SB)
0x0077 00119 (main.go:4) TESTL AX, AX
0x0079 00121 (main.go:4) JNE 139
0x007b 00123 (main.go:7) XCHGL AX, AX
0x007c 00124124 (main.go:7) CALL runtime.deferreturn(SB)
0x0081 00129 (main.go:7) MOVQ 112(SP), BPruntime.deferproc 方法改成了调用 runtime.deferprocStack 方法,难道是做了什么优化?defer 最小单元:_defer
_defer 结构体主要是新增了 heap 字段:type _defer struct {
siz int32
siz int32 // includes both arguments and results
started bool
heap bool
sp uintptr // sp at time of defer
pc uintptr
fn *funcval
..._defer 是在堆上,还是在栈上进行分配,其余字段并没有明确变更,那我们可以把聚焦点放在 defer 的堆栈分配上了,看看是做了什么事。deferprocStack
func deferprocStack(d *_defer) {
gp := getg()
if gp.m.curg != gp {
throw("defer on system stack")
}
d.started = false
d.heap = false
d.sp = getcallersp()
d.pc = getcallerpc()
*(*uintptr)(unsafe.Pointer(&d._panic)) = 0
*(*uintptr)(unsafe.Pointer(&d.link)) = uintptr(unsafe.Pointer(gp._defer))
*(*uintptr)(unsafe.Pointer(&gp._defer)) = uintptr(unsafe.Pointer(d))
return0()
}defer 函数的函数栈指针、传入函数的参数具体地址以及PC(程序计数器),这块在前文 《深入理解 Go defer》 有详细介绍过,这里就不再赘述了。deferprocStack 特殊在哪呢?d.heap 设置为了 false,也就是代表 deferprocStack 方法是针对将 _defer 分配在栈上的应用场景的。deferproc
func newdefer(siz int32) *_defer {
...
d.heap = true
d.link = gp._defer
gp._defer = d
return d
}newdefer 是在哪里调用的呢,如下:func deferproc(siz int32, fn *funcval) { // arguments of fn follow fn
...
sp := getcallersp()
argp := uintptr(unsafe.Pointer(&fn)) + unsafe.Sizeof(fn)
callerpc := getcallerpc()
d := newdefer(siz)
...
}deferproc 方法,现在被用于对应分配到堆上的场景了。小结
deferproc 并没有被去掉,而是流程被优化了。deferproc 还是 deferprocStack 方法,他们分别是针对分配在堆上和栈上的使用场景。优化在哪儿
编译器对 defer 的 for-loop 迭代深度进行分析。// src/cmd/compile/internal/gc/esc.go
case ODEFER:
if e.loopdepth == 1 { // top level
n.Esc = EscNever // force stack allocation of defer record (see ssa.go)
break
}// src/cmd/compile/internal/gc/ssa.go
case ODEFER:
d := callDefer
if n.Esc == EscNever {
d = callDeferStack
}
s.call(n.Left, d)systemstack、mallocgc 等方法所带来的大量性能开销,来达到大部分场景提高性能的作用。循环调用 defer
for-loop 语句等。goto 语句等。显式循环
for 循环中使用 defer 关键字:func main() {
for i := 0; i <= 99; i++ {
defer func() {
fmt.Println("脑子进煎鱼了")
}()
}
}隐式循环
goto 关键字:func main() {
i := 1
food:
defer func() {}()
if i == 1 {
i -= 1
goto food
}
}goto 关键字有时候甚至会被列为代码规范不给使用,主要是会造成一些滥用,所以大多数就选择其实方式实现逻辑。总结
0
0
相关文章
Golang中的Map指针与直接使用Map Go语言Map引用语义解析
如何在Golang中减少系统调用开销 Go语言bufio高效读写优化
解析Golang中的测试并行度控制 Go语言GOMAXPROCS对并发测试的影响
Golang中的缓存预热对基准测试的影响 Go语言冷启动与热启动对比
如何在Golang中优化微服务的内存占用 Go语言高性能RPC框架选型对比
本站声明:本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系admin@php.cn
热门AI工具
相关专题
golang定义变量的方法:1、声明变量并赋予初始值“var age int =值”;2、声明变量但不赋初始值“var age int”;3、使用短变量声明“age :=值”等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。
209
2024.02.23
golang数据转换方法:1、类型转换操作符;2、类型断言;3、字符串和数字之间的转换;4、JSON序列化和反序列化;5、使用标准库进行数据转换;6、使用第三方库进行数据转换;7、自定义数据转换函数。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。
244
2024.02.23
golang常用库有:1、标准库;2、字符串处理库;3、网络库;4、加密库;5、压缩库;6、xml和json解析库;7、日期和时间库;8、数据库操作库;9、文件操作库;10、图像处理库。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。
354
2024.02.23
golang和python的区别是:1、golang是一种编译型语言,而python是一种解释型语言;2、golang天生支持并发编程,而python对并发与并行的支持相对较弱等等。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。
214
2024.03.05
golang是免费的。golang是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的开源编程语言,采用bsd开源协议。本专题为大家提供相关的文章、下载、课程内容,供大家免费下载体验。
407
2024.05.21
本专题围绕 PHP 在现代 Web 后端开发中的高性能实践展开,重点讲解基于 Laravel 框架构建可扩展 API 服务的核心方法。内容涵盖路由与中间件机制、服务容器与依赖注入、接口版本管理、缓存策略设计以及队列异步处理方案。同时结合高并发场景,深入分析性能瓶颈定位与优化思路,帮助开发者构建稳定、高效、易维护的 PHP 后端服务体系。
31
2026.03.04
热门下载
精品课程
最新文章




