golang单元测试性能测试常见用法

9c25ec5ae6cbdb2bb889ff4d4ef4ceda.jpeg

关于go test的一些说明

  • golang安装后可以使用go test工具进行单元测试 代码片段对比的性能测试,使用起来还是比较方便,下面是一些应用场景
    • 平时自己想做一些简单函数的单元测试,不用每次都新建一个main.go 然后go run main.go
    • 相对某个功能做下性能测试 看下cpu/内存消耗情况
  • 可以针对性的建立一些目录 存放自己的一些测试代码
  • 一些使用规则
    • 创建代码保存目录 这里以mytest目录为例
    • 目录下的文件名都要以_test.go结尾 比如common_test.go
    • package mytest 不要为main
    • import testing包
    • 普通单元测试 函数名以Test开头
    • 性能测试 函数名称以Benchmark开头
    • 目录下面函数名不要重名

普通单元测试常见用法

样例代码

  • common_test.go文件内容如下 一个加法函数一个乘法函数
  • package mytest
  • import "testing"

    // Add 加法
    func Add(a int, b int) int {
    return a + b
    }

    // Mul 乘法
    func Mul(a int, b int) int {
    return a * b
    }

    // 测试加法函数功能是否正常 正数
    func TestAddPositive(t *testing.T) {
    if ans := Add(1, 2); ans != 3 {
    t.Fatalf("1 + 2 expected be 3, but %d got", ans)
    }
    t.Log("TestAddPositive run Success")
    }

    // 测试加法函数功能是否正常 负数
    func TestAddNegative(t *testing.T) {
    if ans := Add(-10, -20); ans != -30 {
    t.Fatalf("-10 + -20 expected be -30, but %d got", ans)
    }
    t.Log("TestAddNegative run Success")
    }

    // 测试乘法函数功能是否正常
    func TestMul(t *testing.T) {
    if ans := Mul(3, 4); ans != 12 {
    t.Fatalf("3*4 expected be 12, but %d got", ans)
    }
    t.Log("TestMul run Success")
    }


指定文件测试

  • 命令 顺序测试文件中的所有函数
  • go test -v common_test.go
    • 假设mytest目录下有多个xx_test.go文件 想指定common_test.go文件中的所有函数
  • 输出
  • === RUN TestAddPositive
    common_test.go:20: TestAddPositive run Success
    --- PASS: TestAddPositive (0.00s)
    === RUN TestAddNegative
    common_test.go:28: TestAddNegative run Success
    --- PASS: TestAddNegative (0.00s)
    === RUN TestMul
    common_test.go:36: TestMul run Success
    --- PASS: TestMul (0.00s)
    PASS
    ok command-line-arguments 0.309s


指定测试某一个函数

  • 命令
  • go test -v -run TestAddPositive
    go test -run=TestAddPositive -v
    • 指定只测试目录下的TestAddPositive函数
  • 输出
  • === RUN TestAddPositive
    common_test.go:20: TestAddPositive run Success
    --- PASS: TestAddPositive (0.00s)
    PASS
    ok mytest 0.307s


指定测试某一类函数

  • 命令
  • go test -run=^TestAdd -v
    • 可以使用正则表达式 指定一类函数,比如想测试所有TestAdd开头的函数
  • 输出 会按照顺序测试目录下所有TestAdd打头的函数
  • === RUN TestAddPositive
    common_test.go:20: TestAddPositive run Success
    --- PASS: TestAddPositive (0.00s)
    === RUN TestAddNegative
    common_test.go:28: TestAddNegative run Success
    --- PASS: TestAddNegative (0.00s)
    PASS
    ok mytest 0.359s


测试目录下所有文件中的全部函数

  • 命令 会按照顺序测试目录下所有_test.go结尾的所有函数
  • go test -v .

性能测试的常见用法

样例代码

  • forRangeBenchMark_test.go样例代码如下
  • package mytest

    import "testing"

    // ************ benchmark range loop ************
    type Person struct {
    name [4096]byte
    age int
    }

    var (
    AllPerson [1024]Person
    )

    // BenchmarkForIndexVisit AllPerson[i].xxx 使用下标方式访问
    func BenchmarkForIndexVisit(b *testing.B) {
    for i := 0; i < b.N; i++ {
    var age int
    for i := 0; i < len(AllPerson); i++ {
    age = AllPerson[i].age
    }
    _ = age
    }
    }

    // BenchmarkRangeLoopVisit range循环的方式访问
    func BenchmarkRangeLoopVisit(b *testing.B) {
    for i := 0; i < b.N; i++ {
    var age int
    for _, person := range AllPerson {
    age = person.age
    }
    _ = age
    }
    }


-benchmem会打印内存申请信息,建议都打开这个选项

指定某一个文件测试

  • 命令
  • go test -bench=. -benchmem forRangeBenchMark_test.go
  • 输出
  • goos: darwin
    goarch: amd64
    cpu: Intel(R) Core(TM) i5-9600K CPU @ 3.70GHz
    BenchmarkForIndexVisit-6 4570186 246.8 ns/op 0 B/op 0 allocs/op
    BenchmarkRangeLoopVisit-6 4777 228750 ns/op 0 B/op 0 allocs/op
    PASS
    ok command-line-arguments 2.858s
  • 这里可以看出直接for range的方式遍历性能很差 因为for range会有一个临时变量复制的过程,这个过程比较消耗时间


指定某一个函数测试

  • 命令
  • go test -bench=BenchmarkRangeLoopVisit -benchmem
  • 输出
  • goos: darwin
    goarch: amd64
    pkg: mytest
    cpu: Intel(R) Core(TM) i5-9600K CPU @ 3.70GHz
    BenchmarkRangeLoopVisit-6 4657 345054 ns/op 0 B/op 0 allocs/op
    PASS
    ok mytest 1.950s


指定某一类函数

  • 命令
  • go test -bench=^BenchmarkRange -benchmem
    • 所有以BenchmarkRange打头的都会测试


指定测试时间

  • 命令
  • go test -bench=. -benchmem forRangeBenchMark_test.go -benchtime=5s
    • 默认1s 下面指定运行时间为5s
  • 输出
  • goos: darwin
    goarch: amd64
    cpu: Intel(R) Core(TM) i5-9600K CPU @ 3.70GHz
    BenchmarkForIndexVisit-6 24010926 248.3 ns/op 0 B/op 0 allocs/op
    BenchmarkRangeLoopVisit-6 24183 242653 ns/op 0 B/op 0 allocs/op
    PASS
    ok command-line-arguments 14.918s
    • BenchmarkForIndexVisit函数在5s内运行了24010926次 每次大概耗时 248.3 ns
    • BenchmarkRangeLoopVisit函数在5s内运行了24183次 每次大概耗时 242653 ns 性能相当差


指定测试次数

  • 命令
  • go test -bench=. -benchmem forRangeBenchMark_test.go -benchtime=10000x
  • 输出
  • goos: darwin
    goarch: amd64
    cpu: Intel(R) Core(TM) i5-9600K CPU @ 3.70GHz
    BenchmarkForIndexVisit-6 10000 246.6 ns/op 0 B/op 0 allocs/op
    BenchmarkRangeLoopVisit-6 10000 300624 ns/op 0 B/op 0 allocs/op
    PASS
    ok command-line-arguments 3.347s
    • 测试过程也可以看到BenchmarkRangeLoopVisit运行耗时很长,而BenchmarkForIndexVisit很快就结束了


性能测试-timer相关api

ResetTimer

  • 如果在 benchmark 开始前,需要一些准备工作,如果准备工作比较耗时,则需要将这部分代码的耗时忽略掉

StopTimer & StartTimer

  • 每次函数调用前后需要一些准备工作和清理工作,我们可以使用 StopTimer 暂停计时以及使用 StartTimer 开始计时
  • func Benchmarkxxx(b *testing.B) {
    for n := 0; n < b.N; n++ {
    b.StopTimer()
    // 准备数据
    b.StartTimer()
    Sort(nums)
    }
    }
    func Benchmarkxxx(b *testing.B) {
    //准备工作部分代码
    b.ResetTimer() // 重置定时器
    for n := 0; n < b.N; n++ {
    fib(30) // run fib(30) b.N times
    }
    }

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://xiahunao.cn/news/3247140.html

如若内容造成侵权/违法违规/事实不符,请联系瞎胡闹网进行投诉反馈,一经查实,立即删除!

相关文章

WEB前端05-JavaScrip基本对象

JavaScript对象 1.Function对象 函数的创建 //方法一&#xff1a;自定义函数 function 函数名([参数]) {函数体[return 表达式] }//方法二&#xff1a;匿名函数 (function([参数]) {函数体[return 表达式] }); **使用场景一&#xff1a;定义后直接调用使用(只使用一次) (fun…

RISC-V在线反汇编工具

RISC-V在线反汇编工具&#xff1a; https://luplab.gitlab.io/rvcodecjs/#q34179073&abifalse&isaAUTO 不过&#xff0c;似乎&#xff0c;只支持RV32I、RV64I、RV128I指令集&#xff1a;

Linux·基本指令(下)

1. mv 指令 (move) 语法&#xff1a;mv[选项] 源文件或目录 目标文件或目录 功能&#xff1a;将源文件或目录剪贴到一个新位置&#xff0c;或给源文件或目录改名但不会改变其内容 常用选项&#xff1a; -f &#xff1a;force 强制&#xff0c;如果目标文件已经存在&#xff0c;…

力扣622.设计循环队列

力扣622.设计循环队列 通过数组索引构建一个虚拟的首尾相连的环当front rear时 队列为空当front rear 1时 队列为满 (最后一位不存) class MyCircularQueue {int front;int rear;int capacity;vector<int> elements;public:MyCircularQueue(int k) {//最后一位不存…

vscode+SSH连接Ubuntu

目录 问题引入 基本思路 Permission denied, please try again 修改用户名与密码 新建用户 最终成功方案 问题引入 ssh 用户名ip地址。用户名是远端服务器的用户名&#xff0c;ip地址也是远端服务器的地址。linux虚拟机的ip地址与windous主体不一样&#xff0c;所以还需要…

微信小程序与本地MySQL数据库通信

微信小程序与本地MySQL数据库通信 因为本地MySQL服务器没有域名&#xff0c;也没有进行相应的请求操作封装&#xff0c;因此微信小程序没办法和数据库通信。 但是对于开发人员来说&#xff0c;没有数据库&#xff0c;那还能干撒&#xff1f;虽然我尝试过用json-server&#x…

【C++】类和对象·this指针

C中的类与C语言中的结构体有很多的相似的地方&#xff0c;可以说本质上除了结构体只能定义成员变量&#xff0c;以及结构体默认的访问控制权限是public之外与class没啥区别。但是结构体变量每次调用函数的时候需要指针&#xff0c;而类中的成员函数明明被保存在公共代码段&…

Redis之哈希类型

目录 一.命令 二.内部编码 1.压缩列表&#xff08;ziplist&#xff09; 2. 哈希表&#xff08;Hashtable&#xff09; 自动转换策略 三.作为缓存 Redis的学习专栏&#xff1a;http://t.csdnimg.cn/a8cvV 一.命令 HSET命令 设置hash中指定的字段&#xff08;field)的值,时…

Ubuntu22.04安装OMNeT++

一、官网地址及安装指南 官网地址&#xff1a;OMNeT Discrete Event Simulator 官网安装指南&#xff08;V6.0.3&#xff09;&#xff1a;https://doc.omnetpp.org/omnetpp/InstallGuide.pdf 官网下载地址&#xff1a;OMNeT Downloads 旧版本下载地址&#xff1a;OMNeT Old…

OAI 5G-NR源码架构

OAI 5G-NR源码架构 1 特性范围 目前gNB和5G-NRUE支持如下的配置&#xff1a; 工作模式&#xff1a;TDDCP长度&#xff1a;Normal CP子载波间隔&#xff1a; 30kHz信道带宽&#xff1a;40MHz(106PRB)、80MHz(217PRB)、100MHz(237PRB)天线端口&#xff1a;单波束时隙格式&…

Redis-应用

目录 应用 缓存雪崩、击穿、穿透和解决办法? 布隆过滤器是怎么工作的? 缓存的数据一致性怎么保证 Redis和Mysql消息一致性 业务一致性要求高怎么办? 数据库与缓存的一致性问题 数据库和缓存的一致性如何保证 如何保证本地缓存和分布式缓存的一致&#xff1f; 如果在…

solidity基础语法(以太坊solidity合约)

solidity基础语法&#xff08;以太坊solidity合约&#xff09; 1-值类型和取值范围2-引用类型3-引用类型高阶4-固定数组和动态数组 1-值类型和取值范围 https://learnblockchain.cn/docs/solidity/introduction-to-smart-contracts.html#subcurrency https://learnblockchain…

WEB前端06-DOM对象

BOM浏览器对象模型 浏览器对象模型&#xff1a;将浏览器的各个组成部分封装成对象。是用于描述浏览器中对象与对象之间层次关系的模型&#xff0c;提供了独立于页面内容、并能够与浏览器窗口进行交互的对象结构。 组成部分 Window&#xff1a;浏览器窗口对象 Navigator&…

Hadoop3:HDFS存储优化之小文件归档

一、情景说明 我们知道&#xff0c;NameNode存储一个文件元数据&#xff0c;默认是150byte大小的内存空间。 那么&#xff0c;如果出现很多的小文件&#xff0c;就会导致NameNode的内存占用。 但注意&#xff0c;存储小文件所需要的磁盘容量和数据块的大小无关。 例如&#x…

电脑压缩视频文件 电脑压缩视频大小的方法

在数字化时代&#xff0c;视频已成为我们记录生活、分享快乐的重要工具。然而&#xff0c;大尺寸的视频文件常常让分享和存储变得棘手。如何在保持视频画质的前提下&#xff0c;轻松减小视频文件大小&#xff1f;今天&#xff0c;就让我们一起探索苹果电脑上的几种高效视频压缩…

iPhone手机怎么识别藏文?藏语翻译通App功能介绍:藏文OCR识别提取文字

在工作学习的过程中&#xff0c;遇到不会的藏文&#xff0c;也不知道怎么把文字打出来&#xff0c;这个时候可以试试《藏语翻译通》App的图片识别功能&#xff0c;支持拍照识别和图片识别&#xff0c;拍一拍就能提取藏文文字&#xff0c;并支持一键翻译和复制分享。 跟着小编的…

数据结构之双向链表(赋源码)

数据结构之双向链表(源码) 线性表 双向链表是线性表链式存储结构的一种&#xff0c;若对链式存储结构进行分类可以分为八种。 带头、不带头&#xff1a;指的是该连链表有无头节点&#xff0c;头节点不存放任何内容&#xff0c;它不一定是链表必备的元素&#xff0c;而一个链…

十五、【机器学习】【监督学习】- 神经网络回归

系列文章目录 第一章 【机器学习】初识机器学习 第二章 【机器学习】【监督学习】- 逻辑回归算法 (Logistic Regression) 第三章 【机器学习】【监督学习】- 支持向量机 (SVM) 第四章【机器学习】【监督学习】- K-近邻算法 (K-NN) 第五章【机器学习】【监督学习】- 决策树…

RPA鼠标按键使用技巧

RPA鼠标按键使用技巧 Mouse.MouseAuto.Action命令出错&#xff0c;调用的目标发生了异常&#xff0c;Exception in Mouse.Action元素不可用怎么解决 出现问题 1.想要实现的效果鼠标移动到录屏工具的小球上2.点击开始按钮开始录屏现象&#xff0c;鼠标没有移动痕迹&#xff0c…

C1W4.Assignment.Naive Machine Translation and LSH

理论课&#xff1a;C1W4.Machine Translation and Document Search 文章目录 1. The word embeddings data for English and French words1.1The dataThe subset of dataLoad two dictionaries 1.2 Generate embedding and transform matricesExercise 1: Translating English…