Go 性能优化技巧 3/10

作者: qyuhen | 来源:发表于2016-04-27 10:41 被阅读566次

内置 map 类型是必须的。首先,该类型使用频率很高;其次,可借助 runtime 实现深层次优化(比如说字符串转换,以及 GC 扫描等)。可尽管如此,也不意味着万事大吉,依旧有很多需特别注意的地方。

1预设容量

map 会按需扩张,但须付出数据拷贝和重新哈希成本。如有可能,应尽可能预设足够容量空间,避免此类行为发生。

从结果看,预设容量的 map 显然性能更好,更极大减少了堆内存分配次数。

2 直接存储

对于小对象,直接将数据交由 map 保存,远比用指针高效。这不但减少了堆内存分配,关键还在于垃圾回收器不会扫描非指针类型 key/value 对象。

写个示例对比一下,注意调整 33 行的函数调用。

指针模式:

值模式:

从两次输出里 GC 所占时间百分比,就可看出 “巨大” 差异。

提示:map 对 key、value 数据存储长度有限制。

3 空间收缩

很遗憾,map 不会收缩 “不再使用” 的空间。就算把所有键值删除,它依然保留内存空间以待后用。

就算清空了所有数据,空间依旧没有释放。解决方法是取消 23 行注释,或者替换为一个新的 map 对象。

提示:如长期使用 map 对象(比如用作 cache 容器),偶尔换成 “新的” 或许会更好。还有,int key 要比 string key 更快。

                                                                     ——未完待续


请关注微信公众号:

相关文章

  • Go 性能优化技巧 3/10

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

  • Go 性能优化技巧 10/10

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

  • Go 性能优化技巧 2/10

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

  • Go 性能优化技巧 4/10

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

  • 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 模型。处理双方仅关注通道和数据本身,无需理会对方身份和数量,以此实现结构...

网友评论

  • 戒修:qyunhen 你好:对于go map的key和value的大小限制我有如下的测试代码:

    var s = strings.Repeat("a", 12800000000)

    func main() {
    m := make(map[string]string, 100)
    m[s] = "hello"

    fmt.Println(m[s])
    }

    我在我的机器上是正常的,根据源码的解释, maxKeySize = 128 maxValueSize = 128 的值应该是限制是否能内联处理key和value吧。有点疑惑,麻烦解释下。

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

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