• 企业400电话
  • 微网小程序
  • AI电话机器人
  • 电商代运营
  • 全 部 栏 目

    企业400电话 网络优化推广 AI电话机器人 呼叫中心 网站建设 商标✡知产 微网小程序 电商运营 彩铃•短信 增值拓展业务
    golang 字符串拼接性能的对比分析

    背景

    最近在做一个服务发现/注册的agent, 各个服务需要通过这个agent来注册自己的服务,在完成

    开发后,测试性能时发现性能达不到要求,通过pprof 来确认cpu主要耗费在gc上,分析结果主要是由于字符串拼接导致,故需要测试一下字符串拼接的几种方法的性能;

    字符串拼接的几种方法

    1、直接使用加号进行拼接

    2、strings.Join()

    3、fmt.Sprintf()

    4、bytes.Buffer

    大量字符串拼接性能测试

    我们使用的场景主要是大量字符串拼接,所以需要的场景是不断在字符串上拼接所以测试函数如下:

    // fmt.Printf
    func BenchmarkFmtSprintfMore(b *testing.B) {
     var s string
     for i := 0; i  b.N; i++ {
      s += fmt.Sprintf("%s%s", "hello", "world")
     }
     fmt.Errorf(s)
    }
    // 加号 拼接
    func BenchmarkAddMore(b *testing.B) {
     var s string
     for i := 0; i  b.N; i++ {
      s += "hello" + "world"
     }
     fmt.Errorf(s)
    }
    
    // strings.Join
    func BenchmarkStringsJoinMore(b *testing.B) {
    
     var s string
     for i := 0; i  b.N; i++ {
      s += strings.Join([]string{"hello", "world"}, "")
    
     }
     fmt.Errorf(s)
    }
    
    // bytes.Buffer
    func BenchmarkBufferMore(b *testing.B) {
    
     buffer := bytes.Buffer{}
     for i := 0; i  b.N; i++ {
      buffer.WriteString("hello")
      buffer.WriteString("world")
    
     }
     fmt.Errorf(buffer.String())
    }
    

    执行测试函数

    ~/gopath/src/test/string  go test -bench="."
    goos: darwin
    goarch: amd64
    pkg: test/string
    BenchmarkFmtSprintfMore-4   300000  118493 ns/op
    BenchmarkAddMore-4    300000  124940 ns/op
    BenchmarkStringsJoinMore-4  300000  117050 ns/op
    BenchmarkBufferMore-4   100000000   37.2 ns/op
    PASS
    ok  test/string 112.294s

    从上可以看出使用bytes.buffer的性能是非常高的,如果涉及到大量数据拼接推荐

    bytes.buffer{}

    单次字符串拼接性能测试

    func BenchmarkFmtSprintf(b *testing.B) {
     for i := 0; i  b.N; i++ {
      s := fmt.Sprintf("%s%s", "hello", "world")
      fmt.Errorf(s)
     }
    
    }
    
    func BenchmarkAdd(b *testing.B) {
     for i := 0; i  b.N; i++ {
      s := "hello" + "world"
      fmt.Errorf(s)
     }
    }
    func BenchmarkStringsJoin(b *testing.B) {
     for i := 0; i  b.N; i++ {
      s := strings.Join([]string{"hello", "world"}, "")
      fmt.Errorf(s)
     }
    }
    func BenchmarkBuffer(b *testing.B) {
    
     for i := 0; i  b.N; i++ {
      b := bytes.Buffer{}
      b.WriteString("hello")
      b.WriteString("world")
      fmt.Errorf(b.String())
     }
    }
    

    执行测试函数

     ~/gopath/src/test/string  go test -bench="."
    goos: darwin
    goarch: amd64
    pkg: test/string
    BenchmarkFmtSprintf-4  10000000   200 ns/op
    BenchmarkAdd-4    20000000   93.6 ns/op
    BenchmarkStringsJoin-4  10000000   152 ns/op
    BenchmarkBuffer-4   10000000   175 ns/op
    PASS
    ok  test/string 7.818s

    从上可以看出单词调用字符串拼接性能 + > strings.Join > bytes.Buffer > fmt.Sprintf

    总结

    如果涉及到大量数据拼接推荐 bytes.buffer{}

    后记

    当然只是通过bytes.buffer{} 性能还是不够的,针对这个问题我们通过添加缓存进一步接口qps.

    cpu 耗费在gc 上的原因,需要分析golang gc 回收机制, 这是另外一个topic, 之后总结。

    补充:Go语言字符串批量拼接-StringBuilder

    1.安装

    go get -u github.com/typa01/go-utils

    import (
     "github.com/typa01/go-utils"
    )

    使用,例:

    fieldNames := tsgutils.NewInterfaceBuilder()

    2.使用

    func TestStringBuilderReplace(t *testing.T) {
     builder1 := NewStringBuilder()
     builder1.Append("%111%abc%987%")
     FmtPrintln(builder1.Replace("%", "$").ToString()) // $111$abc$987$
     builder2 := builder1.Clear()
     builder2.AppendStrings("abc","defg").AppendInt(123).AppendFloat64(66.44).AppendStrings("aaa", "bbb", "")
     FmtPrintln(builder2.RemoveLast().ToString()) // abcdefg1236.644E+01aaabbb
     str1 := NewString("123")
     builder3 := NewStringBuilderString(str1).Append("456")
     FmtPrintln(builder3.ToString()) // 123456
    }

    3.GitHub源码地址

    https://github.com/typa01/go-utils

    https://github.com/typa01/go-utils/blob/master/string_builder.go

    以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。如有错误或未考虑完全的地方,望不吝赐教。

    您可能感兴趣的文章:
    • 解决golang时间字符串转time.Time的坑
    • Golang 语言高效使用字符串的方法
    • golang中json小谈之字符串转浮点数的操作
    • golang 如何替换掉字符串里面的换行符\n
    • golang 字符串比较是否相等的方法示例
    • 解决Golang json序列化字符串时多了\的情况
    • golang 获取字符串长度的案例
    • 利用golang的字符串解决leetcode翻转字符串里的单词
    • 用golang如何替换某个文件中的字符串
    上一篇:golang 中的 nil的场景分析
    下一篇:Go语言模型:string的底层数据结构与高效操作详解
  • 相关文章
  • 

    © 2016-2020 巨人网络通讯 版权所有

    《增值电信业务经营许可证》 苏ICP备15040257号-8

    golang 字符串拼接性能的对比分析 golang,字符串,拼接,性能,