Go性能优化技巧 1/10

作者: qyuhen | 来源:发表于2016-04-25 17:59 被阅读806次

字符串(string)作为一种不可变类型,在与字节数组(slice, [ ]byte)转换时需付出 “沉重” 代价,根本原因是对底层字节数组的复制。这种代价会在以万为单位的高并发压力下迅速放大,所以对它的优化常变成 “必须” 行为。

首先,须了解 string 和 [ ]byte 数据结构,并确认默认方式的复制行为。

动态演示: https://asciinema.org/a/6up6gvgqo0v9zkjpusvyucg8g

从 GDB 输出结果可看出,转换后 [ ]byte 底层数组与原 string 内部指针并不相同,以此可确定数据被复制。那么,如不修改数据,仅转换类型,是否可避开复制,从而提升性能?

从 ptype 输出的结构来看,string 可看做 [2]uintptr,而 [ ]byte 则是 [3]uintptr,这便于我们编写代码,无需额外定义结构类型。如此,str2bytes 只需构建 [3]uintptr{ptr, len, len},而 bytes2str 更简单,直接转换指针类型,忽略掉 cap 即可。

用 unsafe 完成指针类型转换,所以得自行为底层数组生命周期做出保证。好在这两个函数都很简单,编译器会完成内联处理,并未发生逃逸行为。

对比一下优化前后的性能差异。

性能提升明显,最关键的是 zero-garbage。

                                                                                                                                                                                   ——未完待续


请关注微信公众号:

相关文章

  • Go性能优化技巧 1/10

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

  • Go 性能优化技巧 10/10

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

  • Go 性能优化技巧 2/10

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

  • Go 性能优化技巧 3/10

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

  • Go 性能优化技巧 4/10

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

  • Go 性能优化技巧 9/10

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

  • Go 性能优化技巧 8/10

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

  • Go 性能优化技巧 7/10

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

  • Go 性能优化技巧 5/10

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

  • Go 性能优化技巧 6/10

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

网友评论

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

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