广告
返回顶部
首页 > 资讯 > 精选 >使用Go defer时要注意什么
  • 664
分享到

使用Go defer时要注意什么

2023-06-20 15:06:21 664人浏览 安东尼
摘要

本篇内容介绍了“使用Go defer时要注意什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!使用 Go defer 要小心这 2 个折腾人

本篇内容介绍了“使用Go defer时要注意什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!

在 Go 语言中 defer 是一个非常有意思的关键字特性。例子如下:

package mainimport "fmt"func main() {    defer fmt.Println("煎鱼了")    fmt.Println("脑子进")}

输出结果是:

脑子进煎鱼了

在前几天我的读者群内有小伙伴讨论起了下面这个问题:

使用Go defer时要注意什么

简单来讲,问题就是针对在 for 循环里搞 defer 关键字,是否会造成什么性能影响

因为在 Go 语言的底层数据结构设计上 defer 是链表的数据结构:

使用Go defer时要注意什么

大家担心如果循环过大 defer 链表会巨长,不够 “精益求精”。又或是猜想会不会 Go defer 的设计和 Redis 数据结构设计类似,自己做了优化,其实没啥大影响?

今天这篇文章,我们就来探索循环 Go defer,造成底层链表过长会不会带来什么问题,若有,具体有什么影响?

开始吸鱼之路。

defer 性能优化 30%

在早年 Go1.13 时曾经对 defer 进行了一轮性能优化,在大部分场景下 提高了 defer 30% 的性能:

使用Go defer时要注意什么

我们来回顾一下 Go1.13 的变更,看看 Go defer 优化在了哪里,这是问题的关键点。

以前和现在对比

在 Go1.12 及以前,调用 Go defer 时汇编代码如下:

    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

在 Go1.13 及以后,调用 Go defer 时汇编代码如下:

    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 00124 (main.go:7)    CALL    runtime.deferreturn(SB)    0x0081 00129 (main.go:7)    MOVQ    112(SP), BP

从汇编的角度来看,像是原本调用 runtime.deferproc 方法改成了调用 runtime.deferprocStack 方法,难道是做了什么优化?

我们抱着疑问继续看下去。

defer 最小单元:_defer

相较于以前的版本,Go 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 并没有被去掉,而是流程被优化了。

  • Go 编译器会根据应用场景去选择使用 deferproc 还是 deferprocStack 方法,他们分别是针对分配在堆上和栈上的使用场景。

优化在哪儿

主要优化在于其 defer 对象的堆栈分配规则的改变,措施是:
编译器对 deferfor-loop 迭代深度进行分析。

// src/cmd/compile/internal/GC/esc.gocase ODEFER:    if e.loopdepth == 1 { // top level        n.Esc = EscNever // force stack allocation of defer record (see ssa.go)        break    }

如果 Go 编译器检测到循环深度(loopdepth)为 1,则设置逃逸分析的结果,将分配到栈上,否则分配到堆上。

// src/cmd/compile/internal/gc/ssa.gocase ODEFER:    d := callDefer    if n.Esc == EscNever {        d = callDeferStack    }    s.call(n.Left, d)

以此免去了以前频繁调用 systemstackmallocgc 等方法所带来的大量性能开销,来达到大部分场景提高性能的作用。

循环调用 defer

回到问题本身,知道了 defer 优化的原理后。那 “循环里搞 defer 关键字,是否会造成什么性能影响?”

最直接的影响就是这大约 30% 的性能优化直接全无,且由于姿势不正确,理论上 defer 既有的开销(链表变长)也变大,性能变差。

因此我们要避免以下两种场景的代码:

  • 显式循环:在调用 defer 关键字的外层有显式的循环调用,例如:for-loop 语句等。

  • 隐式循环:在调用 defer 关键字有类似循环嵌套的逻辑,例如:goto 语句等。

显式循环

第一个例子是直接在代码的 for 循环中使用 defer 关键字:

func main() {    for i := 0; i <= 99; i++ {        defer func() {            fmt.Println("脑子进煎鱼了")        }()    }}

这个也是最常见的模式,无论是写爬虫时,又或是 Goroutine 调用时,不少人都喜欢这么写。

这属于显式的调用了循环。

隐式循环

第二个例子是在代码中使用类似 goto 关键字:

func main() {    i := 1food:    defer func() {}()    if i == 1 {        i -= 1        goto food    }}

这种写法比较少见,因为 goto 关键字有时候甚至会被列为代码规范不给使用,主要是会造成一些滥用,所以大多数就选择其实方式实现逻辑。

这属于隐式的调用,造成了类循环的作用。

总结

显然,Defer 在设计上并没有说做的特别的奇妙。他主要是根据实际的一些应用场景进行了优化,达到了较好的性能。

虽然本身 defer 会带一点点开销,但并没有想象中那么的不堪使用。除非你 defer 所在的代码是需要频繁执行的代码,才需要考虑去做优化。

否则没有必要过度纠结,在实际上,猜测或遇到性能问题时,看看 PProf 的分析,看看 defer 是不是在相应的 hot path 之中,再进行合理优化就好。

所谓的优化,可能也只是去掉 defer 而采用手动执行,并不复杂。在编码时避免踩到 defer 的显式和隐式循环这 2 个雷区就可以达到性能最大化了。

“使用Go defer时要注意什么”的内容就介绍到这里了,感谢大家的阅读。如果想了解更多行业相关的知识可以关注编程网网站,小编将为大家输出更多高质量的实用文章!

--结束END--

本文标题: 使用Go defer时要注意什么

本文链接: https://www.lsjlt.com/news/297766.html(转载时请注明来源链接)

有问题或投稿请发送至: 邮箱/279061341@qq.com    QQ/279061341

本篇文章演示代码以及资料文档资料下载

下载Word文档到电脑,方便收藏和打印~

下载Word文档
猜你喜欢
  • 使用Go defer时要注意什么
    本篇内容介绍了“使用Go defer时要注意什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!使用 Go defer 要小心这 2 个折腾人...
    99+
    2023-06-20
  • Go中defer使用场景及注意事项
    目录1. 简介1.1 使用场景1.2 注意事项2. defer 数据结构3. 执行机制3.1 栈上分配3.2 开放编码4. 参考1. 简介 defer 会在当前函数返回前执行传...
    99+
    2022-06-07
    GO defer
  • python pipeline使用时要注意什么
    这篇文章主要讲解了“python pipeline使用时要注意什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“python pipeline使用时要注意什么”吧!说明在使用之前需要在set...
    99+
    2023-06-20
  • 使用ADO.NET时需要注意什么
    小编给大家分享一下使用ADO.NET时需要注意什么,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!使用ADO.NET时,出现了一个严重的问题,ADO.NET统一了数...
    99+
    2023-06-17
  • DLOG4J在使用MySQL时需要注意什么
    这篇文章主要介绍“DLOG4J在使用MySQL时需要注意什么”,在日常操作中,相信很多人在DLOG4J在使用MySQL时需要注意什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解...
    99+
    2022-10-18
  • Java java.sql.Timestamp时间戳使用要注意什么
    使用Java中的java.sql.Timestamp类表示时间戳时,需要注意以下几点:1. 时间戳是一个特殊的时间数据类型,用于表示...
    99+
    2023-08-09
    Java
  • 使用CSS中的display:none时需要注意什么
    小编给大家分享一下使用CSS中的display:none时需要注意什么,相信大部分人都还不怎么了解,因此分享这篇文章给大家参考一下,希望大家阅读完这篇文章后大有收获,下面让我们一起去了解一下吧!CSS di...
    99+
    2022-10-19
  • 域名使用时需要注意什么问题
    本篇内容主要讲解“域名使用时需要注意什么问题”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“域名使用时需要注意什么问题”吧! 很多人认为在完成域名注册后就可以万事大吉了,实际上并没有那么...
    99+
    2023-06-06
  • 使用http代理ip时​需要注意什么
    这篇文章主要讲解了“使用http代理ip时需要注意什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“使用http代理ip时需要注意什么”吧!1、应该如何选择。市场上有很多这样的软件,一些代理...
    99+
    2023-06-20
  • javascript let关键字使用时要注意什么
    本篇内容主要讲解“javascript let关键字使用时要注意什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“javascript let关键字使用时要注意什么”吧!说明let声明的变量只在...
    99+
    2023-06-20
  • python位置参数使用时要注意什么
    这篇文章主要介绍“python位置参数使用时要注意什么”,在日常操作中,相信很多人在python位置参数使用时要注意什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”python位置参数使用时要注意什么”的疑...
    99+
    2023-06-20
  • 使用sortablejs要注意什么
    这篇文章主要讲解了“使用sortablejs要注意什么”,文中的讲解内容简单清晰,易于学习与理解,下面请大家跟着小编的思路慢慢深入,一起来研究和学习“使用sortablejs要注意什么”吧! 1. 这个方...
    99+
    2022-10-19
  • 使用Go需要注意哪些坑
    本篇内容介绍了“使用Go需要注意哪些坑”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!Go 需要注意的坑需要注意的坑Go 的优势在于方便的协程...
    99+
    2023-06-22
  • java使用new创建对象时要注意什么
    在使用Java的`new`关键字创建对象时,需要注意以下几点:1. 类必须具有公共的构造方法:在使用`new`关键字创建对象时,需要...
    99+
    2023-10-10
    java
  • Java1.5的Enum类型使用时需要注意什么
    本篇内容主要讲解“Java1.5的Enum类型使用时需要注意什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Java1.5的Enum类型使用时需要注意什么”吧!注意点:1。所有创建的枚举类型都...
    99+
    2023-06-03
  • css中类选择器使用时要注意什么
    本篇内容主要讲解“css中类选择器使用时要注意什么”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“css中类选择器使用时要注意什么”吧!注意每个HTML标签都有一个属性叫做class, 也就是说每...
    99+
    2023-06-20
  • css中id选择器使用时要注意什么
    这篇文章主要介绍“css中id选择器使用时要注意什么”,在日常操作中,相信很多人在css中id选择器使用时要注意什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”css中id选择器使用时要注意什么”的疑惑有所...
    99+
    2023-06-20
  • css标签选择器使用时要注意什么
    本篇内容介绍了“css标签选择器使用时要注意什么”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!作用: 根据指定的标签名称, 在当前界面中找到...
    99+
    2023-06-20
  • 注册域名时要注意什么问题
    本篇内容介绍了“注册域名时要注意什么问题”的有关知识,在实际案例的操作过程中,不少人都会遇到这样的困境,接下来就让小编带领大家学习一下如何处理这些情况吧!希望大家仔细阅读,能够学有所成!第一点,域名后缀名。咱们都明白,最为盛行的域名后缀是C...
    99+
    2023-06-06
  • DIV+CSS布局时要注意什么
    这篇文章主要介绍“DIV+CSS布局时要注意什么”,在日常操作中,相信很多人在DIV+CSS布局时要注意什么问题上存在疑惑,小编查阅了各式资料,整理出简单好用的操作方法,希望对大家解答”DIV+CSS布局时...
    99+
    2022-10-19
软考高级职称资格查询
编程网,编程工程师的家园,是目前国内优秀的开源技术社区之一,形成了由开源软件库、代码分享、资讯、协作翻译、讨论区和博客等几大频道内容,为IT开发者提供了一个发现、使用、并交流开源技术的平台。
  • 官方手机版

  • 微信公众号

  • 商务合作