Golang 性能基准测试(benchmark)详解

 更新时间:2023年06月04日 10:02:46   作者:路多辛  
Golang性能基准测试可以帮助开发人员比较不同的实现方式对性能的影响,以便优化程序,本文就来讲解一下如何使用Golang的性能基准测试功能,需要的朋友可以参考下

Golang 性能基准测试

Golang 中的性能基准测试是使用标准库 testing 来实现的,编写性能测试代码是很容易的:

  • 创建性能测试文件:在 Go 项目的源代码目录下创建一个新的文件(和被测代码文件在同一个包),以 _test.go 为后缀名。例如,要测试net包中 dial.go 中的方法,在 net 包中创建一个名字为 dial_test.go 文件,和单元测试文件是一样的。
  • 导入 testing 包:在测试文件中导入testing包,以使用相关的的函数和工具。
  • 编写测试函数:在测试文件中,编写一个以 Benchmark 为前缀的函数,后面跟上一个或多个字符或字符组合来标识测试用例的名称(一般使用被测的函数名称),参数必须是 b *testing.B。
  • 编写测试代码:b.N是基准测试框架提供的,表示循环的次数,因为需要反复调用测试代码来评估性能。b.N 的值会以1, 2, 5, 10, 20, 50, …这样的规律递增下去直到运行时间大于1秒钟,由于程序判断运行时间稳定才会停止运行,所以千万不要在loop循环里面使用一个变化的值作为函数的参数。

以 json 格式校验工具
https://github.com/luduoxin/json-validator-go 为例,validator包中的 scanner.go 文件中的关键函数 Valid 用于校验给定字符串是否 json 格式,对应的性能测试文件为 scanner_test.go,里面的测试函数为 BenchmarkValid,代码如下:

package validator
import "testing"
func BenchmarkValid(b *testing.B) {
	str := `{"foo":"bar"}`
	b.ResetTimer()
	for i := 0; i < b.N; i++ {
		Valid([]byte(str))
	}
}

运行性能测试用例

性能测试命令为 go test [参数],比如 go test -bench=. ,具体的命令参数及含义如下:

-bench regexp 性能测试,运行指定的测试函

-bench . 运行所有的benchmark函数测试,指定名称则只执行具体测试方法而不是全部

-benchmem 性能测试的时候显示测试函数的内存分配的统计信息

-count n 运行测试和性能多少此,默认一次

-run regexp 只运行特定的测试函数

-timeout t 测试时间如果超过 t 则panic,默认10分钟

-v 显示测试的详细信息

启动命令行,切换到 json-validator-go 项目的 validator 文件夹下,运行全部性能测试用例:

$ go test -bench=.
goos: darwin
goarch: amd64
pkg: github.com/luduoxin/json-validator-go/validator
cpu: Intel(R) Core(TM) i5-8259U CPU @ 2.30GHz
BenchmarkValid-8        13562608                86.55 ns/op
PASS
ok      github.com/luduoxin/json-validator-go/validator 1.420s

上面输出的报告的倒数第三行信息的内容含义如下:

BenchmarkValid 是性能测试函数名称,-8 表示 GOMAXPROCS 的值为8,13562608 表示一共执行了13562608次,即b.N的值,86.55 ns/op 表示平均每次操作花费了 86.55 纳秒。

在一个测试方法里面也可以跑多个用例,使用更多的类型的数据分别看下对应的性能,代码如下:

package validator
import "testing"
func BenchmarkValid(b *testing.B) {
	var validTests = []struct {
		data string
		ok   bool
	}{
		{`foo`, false},
		{`}{`, false},
		{`{]`, false},
		{`{}`, true},
		{`[{}]`, true},
		{`{"foo":"bar"}`, true},
		{`{"foo":"bar","bar":{"baz":["qux"]}}`, true},
	}
	for _, v := range validTests {
		b.Run("", func(b *testing.B) {
			for i := 0; i < b.N; i++ {
				Valid([]byte(v.data))
			}
		})
	}
}

运行看下效果:

$ go test -bench=. 
goos: darwin
goarch: amd64
pkg: github.com/luduoxin/json-validator-go/validator
cpu: Intel(R) Core(TM) i5-8259U CPU @ 2.30GHz
BenchmarkValid/#00-8             4746290               249.8 ns/op
BenchmarkValid/#01-8             4841005               245.5 ns/op
BenchmarkValid/#02-8             4610671               257.0 ns/op
BenchmarkValid/#03-8            26957421                42.63 ns/op
BenchmarkValid/#04-8            29747263                41.88 ns/op
BenchmarkValid/#05-8            20895832                56.31 ns/op
BenchmarkValid/#06-8            14058906                83.17 ns/op
BenchmarkValid/#07-8             5518412               212.9 ns/op
PASS
ok      github.com/luduoxin/json-validator-go/validator 10.891s

到此这篇关于Golang 性能基准测试(benchmark)详解的文章就介绍到这了,更多相关Golang 性能基准测试内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 在goland中读取tpl文件的图文操作

    在goland中读取tpl文件的图文操作

    这篇文章主要介绍了在goland中读取tpl文件的图文操作,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-12-12
  • 学会提升Go语言编码效率技巧拒绝加班!

    学会提升Go语言编码效率技巧拒绝加班!

    这篇文章主要为大家介绍了Go语言编码效率提升技巧详解,学会了从此拒绝加班,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2023-12-12
  • RabbitMQ延时消息队列在golang中的使用详解

    RabbitMQ延时消息队列在golang中的使用详解

    延时队列常使用在某些业务场景,使用延时队列可以简化系统的设计和开发、提高系统的可靠性和可用性、提高系统的性能,下面我们就来看看如何在golang中使用RabbitMQ的延时消息队列吧
    2023-11-11
  • 一文带你玩转Golang Prometheus Eexporter开发

    一文带你玩转Golang Prometheus Eexporter开发

    本文分两大块,一是搞清楚prometheus四种类型的指标Counter,Gauge,Histogram,Summary用golang语言如何构造这4种类型对应的指标,二是搞清楚修改指标值的场景和方式,感兴趣的可以了解一下
    2023-02-02
  • go切片和指针切片示例详解

    go切片和指针切片示例详解

    在Go语言中,切片(Slice)和指针的切片(即切片中每个元素都是指向某种数据类型的指针)是两个不同的概念,它们各自具有特定的用途和优势,这篇文章主要介绍了go切片和指针切片,需要的朋友可以参考下
    2024-04-04
  • 关于go语言编码需要放到src 文件夹下的问题

    关于go语言编码需要放到src 文件夹下的问题

    这篇文章主要介绍了go语言编码需要放到src 文件夹下的相关知识,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2020-10-10
  • 一文带你了解Golang中的并发性

    一文带你了解Golang中的并发性

    并发是一个很酷的话题,一旦你掌握了它,就会成为一笔巨大的财富。所以本文就来和大家一起来聊聊Golang中的并发性,感兴趣的可以了解一下
    2023-03-03
  • Golang中的四个括号示例详解

    Golang中的四个括号示例详解

    这篇文章主要介绍了Golang中的四个括号,本文通过实例代码给大家介绍的非常详细,通过实例代码补充介绍了有效的括号golang实现,需要的朋友可以参考下
    2024-03-03
  • Golang通道channel的源码分析

    Golang通道channel的源码分析

    channel(通道),顾名思义,是一种通道,一种用于并发环境中数据传递的通道。channel是golang中标志性的概念之一,很好很强大!本文将从源码带大家了解一下channel的使用,希望对大家有所帮助
    2022-12-12
  • Golang获取当前时间代码

    Golang获取当前时间代码

    本文给大家汇总介绍了golang中的相关的时间的操作,有需要的小伙伴可以拿走参考下
    2018-10-10

最新评论