首页/ 文章/后端/正文


前言

单元测试是用来对一个模块、一个函数或者一个类来进行正确性检验的测试工作。我们根据它来检验代码的行为是否和预期的一样,如果单元测试不通过,要么代码有bug,要么测试条件输入不正确,总之,需要修复使单元测试能够通过。单元测试一个最大的好处,就是确保一个程序模块的行为符合我们设计的预期,在将来对代码进行修改/重构时,还能最大限度地保证代码的行为仍然正确。

Go对单元测试的支持已经相当友好了,原生的go test标准库就是专门用来进行单元测试的编写的。使用go test编写单元测试时需要遵循一些约定,比如所有测试代码都需要添加到_test.go结尾的测试文件中,这样在使用go build进行构建时,测试代码才会被排除在外。另外,每个测试函数都必须导入testing包,测试函数的名字必须以Test开头,且跟在Test后面的后缀名必须以大写开头,因此测试函数的声明应该是这样的:

  func TestSin(t *testing.T) { /* ... */ }   func TestCos(t *testing.T) { /* ... */ }   func TestLog(t *testing.T) { /* ... */ } 复制代码

_test.go测试文件通常和需要被测试的文件放在同一个包内,比如有如下的一段待测试的代码(在word包目录下的word.go文件):

package word // 判断一个字符串s是否时回文字符串 func IsPalindrome(s string) bool {     for i := range s {         if s[i] != s[len(s)-1-i] {             return false         }      }     return true } 复制代码

那么在编写测试时,我们同样在word包目录下,创建一个word_test.go文件,单元测试的代码如下:

package word import "testing" func TestPalindrome(t *testing.T) {         // "detartrated"是一个回文字符串,因此IsPalindrome("detartrated")的返回值应该为true         // 如果返回false,则表示其实现有问题,需要使用t.Error进行错误报告 if !IsPalindrome("detartrated") {     t.Error(`IsPalindrome("detartrated") = false`) } if !IsPalindrome("kayak") {     t.Error(`IsPalindrome("kayak") = false`) } } func TestNonPalindrome(t *testing.T) {     if IsPalindrome("palindrome") {         t.Error(`IsPalindrome("palindrome") = true`)      } } 复制代码

测试用例成功则打印:

=== RUN   TestPalindrome --- PASS: TestPalindrome (0.00s) PASS 复制代码

用例执行失败则是:

=== RUN   TestNonPalindrome --- FAIL: TestNonPalindrome (0.00s)     nba_test.go:53: IsPalindrome("kayak") = true FAIL 复制代码

没有断言的go test框架

从单元测试的代码中可以看出,在判断IsPalindrome方法是否运行正确时,我们并未使用断言,而是通过if语句进行判断:如果结果错误则通过t.Error方法报告错误。这是因为Go原生就不支持断言,官网也给出了他们这样设计的原因,简单来说就是不想让开发者在错误处理上进行偷懒

Go doesn't provide assertions. They are undeniably convenient, but our experience has been that programmers use them as a crutch to avoid thinking about proper error handling and reporting...(详见https://golang.org/doc/faq#assertions)

但对单元测试来说,没有断言真的是很不方便。特别地,对以前做Java开发时习惯了使用Junit进行单元测试的同学而言更是难受。另外,使用if语句来对方法的输出结果进行判断,并进行对应的错误处理,会导致单元测试代码充斥着大量的if分支,影响了代码的可读性

那么,有没有更好的解决方法呢?

针对这个问题,我们可以引入第三方的断言库来解决。

使用testify进行断言

在第三方断言库的选择上,就活跃度和易用性而言,testify都是最佳的选择。

testify:https://github.com/stretchr/testify

现在,我们使用testify为上述IsPalindrome的单元测试用例进行重构:

package word import "testing" import "github.com/stretchr/testify/assert" func TestPalindrome(t *testing.T) {     // 断言IsPalindrome方法的返回值为True     assert.True(t, IsPalindrome("detartrated"))     assert.True(t, IsPalindrome("kayak")) } func TestNonPalindrome(t *testing.T) {     // 断言IsPalindrome方法的返回值为False     assert.False(t, IsPalindrome("palindrome")) } 复制代码

测试用例成功则打印:

=== RUN   TestPalindrome --- PASS: TestPalindrome (0.00s) PASS 复制代码

用例执行失败则是:

=== RUN   TestNonPalindrome     palindrome_test.go:23:           Error Trace: palindrome_test.go:23          Error:       Should be false          Test:        TestNonPalindrome --- FAIL: TestNonPalindrome (0.00s) 复制代码

由此可见,重构后的测试用例更加简洁,可读性也更加好了。而且,在断言出错的情况下,打印出来的错误信息也相对丰富。

testify底层实现也是基于go test框架,在上述用法中,每个assert方法都将testing.T作为第一个入参,以此来使用go test框架的错误报告的基础能力。另外,如果你在一个测试方法中需要断言很多次,每次都传参testing.T就显得比较繁琐,那么可以这样实现来简化:

func TestSomething(t *testing.T) {     // 创建一个assert实例,只需传参testing.T一次     assert := assert.New(t)     // 断言两者相等     assert.Equal(123, 123)     // 断言两者不相等     assert.NotEqual(123, 456)     // 断言某个实例为nil     assert.Nil(object)     // 断言object实例不为nil     if assert.NotNil(object) {         // 当知道object不为nil之后,就可以安全地对其进行访问了         // 进一步断言object的Value属性的值为"Something"         assert.Equal("Something", object.Value)     } } 复制代码

testify有着极其丰富的断言方法,除了上述几个assert.Trueassert.Nilassert.Equal常用的断言方法以外,还有用来断言目录是否存在的assert.DirExists;断言某个方法是否抛出panicassert.Panics;断言字符串是否符合指定正则表达式的assert.Regexp,等等。总之,testify绝对能够满足你在Go单元测试断言方面的所有需要。

更多用法请参考testify assert API文档:https://godoc.org/github.com/stretchr/testify/assert

总结

本文主要介绍了如何使用testify库来为Go单元测试引入断言的能力,以此来使得测试用例代码更加简洁。assert只是testify库提供能力之一,比如它还有mock包,提供了“打桩”的能力。说起“打桩”,写过Java单元测试的同学一定很熟悉,Java中的MockitoPowerMock框架就提供了类似的功能。本文之所以没有介绍testify库的mock功能,是因为还有更加好用的mock库。下一篇文章,我们将介绍如何高效、优雅地在Go单元测试中进行打桩。


打赏

好文章,更需要你的鼓励

免责声明:本文转载至互联网,不代表本站的观点和立场。

浏览次数:53 次浏览

发布时间:2020-12-25 12:44:47

相关标签: gogolang单元测试

本文链接:http://pay.21stf.org/show-88.html