登录
首页 >  Golang >  Go问答

如何在 Go 中通过反射测试函数集合?

来源:Golang技术栈

时间:2023-05-03 17:02:00 257浏览 收藏

欢迎各位小伙伴来到golang学习网,相聚于此都是缘哈哈哈!今天我给大家带来《如何在 Go 中通过反射测试函数集合?》,这篇文章主要讲到golang等等知识,如果你对Golang相关的知识非常感兴趣或者正在自学,都可以关注我,我会持续更新相关文章!当然,有什么建议也欢迎在评论留言提出!一起学习!

问题内容

我必须为具有相似签名和返回值(一个对象和一个错误)的几个函数编写单元测试,这些函数必须通过相似的测试条件。
我想避免写:

func TestFunc1(t *testing.T) {
    // tests on return values
}
func TestFunc2(t *testing.T) {
    // tests identical for Func1
}
func TestFunc3(t *testing.T) {
    // tests identical for Func1
}
...

(有关更完整的上下文,请参阅此 go playground 示例
) (是的,go playground 尚不支持go test,仅go run,并且issue 6511是为了请求该功能)

您将如何使用反射(reflectpackage)来只编写一个测试:

  • 依次调用每个函数?
  • 测试它们的返回值?

我见过:

  • “如何.Call在反射包中正确使用,Golang? ”,使用Value.Call
  • “从 Golang 中的函数列表中选择一个函数”

但我错过了调用函数和在测试中使用返回值的完整示例。

正确答案

一旦我明白一切都必须使用或返回类型 Value,这就是我想出的。
诀窍是使用:

测试代码的主要摘录:

var funcNames = []string{"Func1", "Func2", "Func3"}

func TestFunc(t *testing.T) {
    stype := reflect.ValueOf(s)
    for _, fname := range funcNames {

        fmt.Println(fname)

        sfunc := stype.MethodByName(fname)
        // no parameter => empty slice of Value
        ret := sfunc.Call([]reflect.Value{})

        val := ret[0].Int()

        // That would panic for a nil returned err
        // err := ret[1].Interface().(error)
                err := ret[1]

        if val 

请参阅go playground 中的可运行示例


请注意,如果您使用不存在的函数名调用该测试函数,则会出现恐慌。
请参阅此处的示例

runtime.panic(0x126660, 0x10533140)
    /tmp/sandbox/go/src/pkg/runtime/panic.c:266 +0xe0
testing.func路005()
    /tmp/sandbox/go/src/pkg/testing/testing.go:383 +0x180
----- stack segment boundary -----
runtime.panic(0x126660, 0x10533140)
    /tmp/sandbox/go/src/pkg/runtime/panic.c:248 +0x160
reflect.flag.mustBe(0x0, 0x13)
    /tmp/sandbox/go/src/pkg/reflect/value.go:249 +0xc0
reflect.Value.Call(0x0, 0x0, 0x0, 0xfeef9f28, 0x0, ...)
    /tmp/sandbox/go/src/pkg/reflect/value.go:351 +0x40
main.TestFunc(0x10546120, 0xe)
    /tmpfs/gosandbox-3642d986_9569fcc1_f443bbfb_73e4528d_c874f1af/prog.go:34 +0x240

去操场从那种恐慌中恢复过来,但你的测试程序可能不会。

这就是为什么我添加到上面的测试功能:

for _, fname := range funcNames {

    defer func() {
        if x := recover(); x != nil {
            t.Error("TestFunc paniced for", fname, ": ", x)
        }
    }()
    fmt.Println(fname)

这会产生(参见示例)更好的输出:

Func1
Func2
Func3
Func4
--- FAIL: TestFunc (0.00 seconds)
    prog.go:48: Func2 should return positive value
    prog.go:51: Func3 shouldn't err
    prog.go:32: TestFunc paniced for Func4 :  reflect: call of reflect.Value.Call on zero Value
FAIL

理论要掌握,实操不能落!以上关于《如何在 Go 中通过反射测试函数集合?》的详细介绍,大家都掌握了吧!如果想要继续提升自己的能力,那么就来关注golang学习网公众号吧!

声明:本文转载于:Golang技术栈 如有侵犯,请联系study_golang@163.com删除
相关阅读
更多>
最新阅读
更多>
课程推荐
更多>