Go 性能优化技巧 4/10

作者: qyuhen | 来源:发表于2016-04-28 11:05 被阅读206次

延迟调用(defer)确实是一种 “优雅” 机制。可简化代码,并确保即便发生 panic 依然会被执行。如将 panic/recover 比作 try/except,那么 defer 似乎可看做 finally。

如同异常保护被滥用一样,defer 被无限制使用的例子比比皆是。

只需稍稍了解 defer 实现机制,就不难理解会有这样的性能差异。

编译器通过 runtime.deferproc “注册” 延迟调用,除目标函数地址外,还会复制相关参数(包括 receiver)。在函数返回前,执行 runtime.deferreturn 提取相关信息执行延迟调用。这其中的代价自然不是普通函数调用一条 CALL 指令所能比拟的。

或许你会觉得 4x 的性能差异算不得什么,但如果是下面这样呢?

当多个 goroutine 执行该函数时,只怕性能差异就不是 4x,还得算上 httpGet 所需时间。原本的并发设计,因为错误的 defer 调用变成 “串行”。

与之类似的,还有下面这样的写法。

如果 files 是个 “超大” 列表,只怕在 analysis 结束前,会有不小的隐式 “资源泄露”,这些不能及时回收的对象,会导致 GC 在内的相关性能问题。

解决方法么,要么去掉 f.close 前的 defer,要么将内层处理逻辑重构为独立函数(比如匿名函数调用)。

除此之外,单个函数里过多的 defer 调用可尝试合并。最起码,在并发竞争激烈时,mutex.Unlock 不应该使用 defer,而应尽快执行,仅保护最短的代码片段。

——未完待续


请关注微信公众号

相关文章

  • Go 性能优化技巧 4/10

    延迟调用(defer)确实是一种 “优雅” 机制。可简化代码,并确保即便发生 panic 依然会被执行。如将 pa...

  • Go 性能优化技巧 10/10

    垃圾回收不是万能的,Go 一样存在资源泄露问题。 1SetFinalizer 虽然垃圾回收器能很好地处理循环引用,...

  • Go 性能优化技巧 2/10

    对于一些初学者,自知道 Go 里面的 array 以 pass-by-value 方式传递后,就莫名地引起 “恐慌...

  • Go 性能优化技巧 3/10

    内置 map 类型是必须的。首先,该类型使用频率很高;其次,可借助 runtime 实现深层次优化(比如说字符串转...

  • Go 性能优化技巧 9/10

    作为内置类型,通道(channel)从运行时得到很多支持,其自身设计也算得上精巧。但不管怎么说,它本质上依旧是一种...

  • Go性能优化技巧 1/10

    字符串(string)作为一种不可变类型,在与字节数组(slice, [ ]byte)转换时需付出 “沉重” 代价...

  • Go 性能优化技巧 8/10

    尽管反射(reflect)存在性能问题,但依然被频繁使用,以弥补静态语言在动态行为上的不足。只是某些时候,我们须对...

  • Go 性能优化技巧 7/10

    接口的用途无需多言。但这并不意味着可在任何场合使用接口,要知道通过接口调用和普通调用存在很大差别。首先,相比静态绑...

  • Go 性能优化技巧 5/10

    闭包(closure)也是很常见的编码模式,因它隐式携带上下文环境变量,因此可让算法代码变得更加简洁。 但任何 “...

  • Go 性能优化技巧 6/10

    Go 使用 channel 实现 CSP 模型。处理双方仅关注通道和数据本身,无需理会对方身份和数量,以此实现结构...

网友评论

    本文标题:Go 性能优化技巧 4/10

    本文链接:https://www.haomeiwen.com/subject/psqhrttx.html