本文介绍golang 如何做基准性能测试。

编写完代码除了跑必要的单元测试外,还需要考虑代码跑起来的性能如何。性能的衡量其实就是程序运行时候进程的内存分配,CPU消耗情况。

golang 语言在提供了功能测试的基础上,提供了丰富的性能测试功能。

SHOW CODE

首先,从一个例子来讲起。 随便写一个简单的快速排序,然后和系统自带的排序做一个性能比较。

如下为简版快排的代码:

  1. package benchmark
  2. import "sort"
  3. func QSort(data []int) {
  4. myqsort(data, 0, len(data)-1)
  5. }
  6. func myqsort(data []int, s, e int) {
  7. if s >= e {
  8. return
  9. }
  10. t := data[s]
  11. i, j := s, e
  12. for i < j {
  13. for ; i < j && data[j] >= t; j-- { }
  14. for ; i < j && data[i] < t; i++ { }
  15. if i < j { break }
  16. data[i], data[j] = data[j], data[i]
  17. i++
  18. j--
  19. }
  20. data[i] = t
  21. myqsort(data, s, i-1)
  22. myqsort(data, i+1, e)
  23. }

然后编写一个测试的test。

  1. package benchmark
  2. import "testing"
  3. import "math/rand"
  4. import "time"
  5. import "sort"
  6. var ints []int
  7. // 长度为 1w 的数据使用系统自带排序
  8. func BenchmarkSort10k(t *testing.B) {
  9. slice := ints[0:10000]
  10. t.ResetTimer() // 只考虑下面代码的运行事件,所以重置计时器
  11. for i := 0; i < t.N; i++ {
  12. sort.Ints(slice)
  13. }
  14. }
  15. // 长度为 100 的数据使用系统自带排序
  16. func BenchmarkSort100(t *testing.B) {
  17. slice := ints[0:100]
  18. t.ResetTimer()
  19. for i := 0; i < t.N; i++ {
  20. sort.Ints(slice)
  21. }
  22. }
  23. // 长度为 1w 的数据使用上述代码排序
  24. func BenchmarkQsort10k(t *testing.B) {
  25. slice := ints[0:10000]
  26. t.ResetTimer()
  27. for i := 0; i < t.N; i++ {
  28. QSort(slice)
  29. }
  30. }
  31. // 长度为 100 的数据使用上述代码排序
  32. func BenchmarkQsort100(t *testing.B) {
  33. slice := ints[0:100]
  34. t.ResetTimer()
  35. for i := 0; i < t.N; i++ {
  36. QSort(slice)
  37. }
  38. }
  39. // 数据初始化,为了保证每次数据都是一致的。
  40. func TestMain(m *testing.M) {
  41. rand.Seed(time.Now().Unix())
  42. ints = make([]int, 10000)
  43. for i := 0; i < 10000; i++ {
  44. ints[i] = rand.Int()
  45. }
  46. m.Run()
  47. }

运行命令 :

  1. # go test -cover -count 3 -benchmem -bench=.

运行结果如下图:

基准测试,默认将每个方法执行1s中,然后展示执行的次数,每一次执行的耗时, 上述还展示了内存每次分配的大小,以及每次benchmark分配的次数。上述的命令行指定了运行次数为3次,显示代码覆盖率和内存分配情况。

从基准测试的结果可以分析出:对于1w数据量的排序,自带的排序比我的排序算法要快20倍左右;100数据量的排序,手撸的排序略胜一筹。

从内存分析来讲,系统自带的会使用4B的数据,而我的算法无内存分配。

INTRODUCE BENCHMARK

引入golang 提供的 testing 包,写需要的基准测试的方法(方法名必须以Benchmark开头, 参数必须为 *testing.B)。

若需要做一些数据初始化的工作,可以如上写一个TestMain 方法,将数据初始化的工作在这里完成。

除了这些,可以看*testing.B, *testing.M 的相关方法即可。

最后,只要运行官方提供的 go test -bench=. 命令,即可开始跑基准测试。 当然,还有其他选项可以满足我们多样的需求。

例如:

  • -cpu 1,2,4 指定运行的cpu 格式
  • -count n 指定运行的次数
  • -benchtime 每一条测试执行的时间 (默认是1s)
  • -bench 指定执行bench的方法, . 是全部
  • -benchmem 显示内存分配情况

其他参数可以通过 go help testflag 查看

WHY SO SLOW

  1. 我这里选取的是第一个数作为中位数,数据越大越可能出现倾斜,排序慢的概率也大。
  2. 正常的排序包中,都会在对小于等于12 个数的数组做排序时使用希尔排序,速度也有很大提升。

除了简单的做性能测试外,golang 还自带了性能分析的工具,我们可以快速找出代码中的内存分配、cpu消耗的核心区,帮助我们解决服务的性能问题。下篇文章将做详细了解。

golang 性能测试 (1)的更多相关文章

  1. golang 性能测试pprof

    golang 性能测试包是位于 net/http 包下的 pprof,其相关介绍可以参看具体的 官方文档 有关 golang 性能测试使用特别简单,在 main 包中的引包位置直接引入: import ...

  2. Golang 性能测试 (3) 跟踪刨析 golang trace

    简介 对于绝大部分服务,跟踪刨析是用不到的.但是如果遇到了下面问题,可以不妨一试: 怀疑哪个协程慢了 系统调用有问题 协程调度问题 (chan 交互.互斥锁.信号量等) 怀疑是 gc (Garbage ...

  3. Golang 语言的单元测试和性能测试(也叫 压力测试)

    Golang单元测试对文件名和方法名,参数都有很严格的要求. 例如: 1.文件名必须以xx_test.go命名 2.方法必须是Test[^a-z]开头(T必须大写),func TestXxx (t * ...

  4. golang bufio、ioutil读文件的速度比较(性能测试)和影响因素分析

    前言 golang读取文件的方式主要有4种: 使用File自带的Read方法 使用bufio库的Read方法 使用io/ioutil库的ReadAll() 使用io/ioutil库的ReadFile( ...

  5. 基本http服务性能测试(Python vs Golang)

    最近学习Golang,总想体验下并发到底有多叼,必我大 python强势多少. 学习了官方教程的http 服务,用性能测试工具wrk测试了下,发现结果很令人惊讶- wrk可以参考我的博客,有基本用法说 ...

  6. Golang RPC 性能测试

    Golang RPC 性能测试 | KDF5000 http://kdf5000.com/2017/03/28/Golang-RPC-性能测试/

  7. golang 单元测试&&性能测试

    一:单元测试 1.为什么要做单元测试和性能测试 减少bug 快速定位bug 减少调试时间 提高代码质量 2.golang的单元测试 单元测试代码的go文件必须以_test.go结尾 单元测试的函数名必 ...

  8. LevelDB性能测试|Golang调用LevelDB

    LevelDB性能测试|Golang调用LevelDB 不同方式使用压力测试 用ssdb,TCP连接方式调用,底层存储levelDB 直接调用Cgo的levelDB (必须保证串行) 直接调用Gola ...

  9. 【GoLang】GoLang 单元测试、性能测试使用方法

    单元测试代码: ackage test import ( // "fmt" "testing" ) func Test_FlowControl(t *testi ...

随机推荐

  1. celery异步任务 定时任务

    以前项目中用到过 celery ,但是没怎么记笔记,现在在记一下,方便以后用.   Celery.png 问:Celery 是什么? 答:Celery 是一个由 Python 编写的简单.灵活.可靠的 ...

  2. react 新创建项目

    1,先创建一个文件夹用于存放项目 2,运行cmd,路径选择到你创建的文件夹内 3,  npm install -g create-react-app         create-react-app ...

  3. OPENGL图形渲染管线图解

    OPENGL固定图形渲染管线可以粗略地认为由下面的阶段衔接而成: 顶点颜色,光照,材质三个输入在光栅化前控制绘制管线的操作.光照和材质不能单独使用.顶点颜色,光源颜色,材质颜色都有alpha值,它们的 ...

  4. 前端构建工具gulpjs的使用介绍及技巧【转载】

    转载至:http://www.cnblogs.com/2050/p/4198792.html gulpjs是一个前端构建工具,与gruntjs相比,gulpjs无需写一大堆繁杂的配置参数,API也非常 ...

  5. 快速上手项目远程团队协作--CODING(新手向)

    快速上手项目远程团队协作--CODING(新手向) CODING网址:https://e.coding.net/signin 前言:之前用过github,用过微软的Azure协作(https://az ...

  6. 推荐两款好用的JS格式化工具

    工具一: 直接在Chrome浏览器中,F12,打开Sources栏,找到JS文件,点击下面的花括号即可. 工具二: 使用notepad++ 格式化JS文件. 1.下载 jstool 插件(https: ...

  7. Java8 Stream流

    第三章 Stream流 <Java8 Stream编码实战>的代码全部在https://github.com/yu-linfeng/BlogRepositories/tree/master ...

  8. json到底是什么??????

    JSON(JavaScript Object Notation)是一种基于JavaScript语法子集的开放标准数据交换格式.JSON是基于文本的,轻量级的,通常被认为易于读/写. 通俗解释: 1.j ...

  9. Untargeted lipidomics reveals specific lipid abnormality in nonfunctioning human pituitary adenomas 非靶向脂质组学揭示非功能人类脑垂体瘤中的特异性脂质 (解读人:胡丹丹)

    文献名:Untargeted lipidomics reveals specific lipid abnormality in nonfunctioning human pituitary adeno ...

  10. xheditor图片上传

    前端步骤:分为三部,这三部做完之后就能正确显示富文本了 1.下载xheditor文件,并按照如下要求进行引入: <!-- xheditor富文本的文件引入 --> <script t ...