当前位置: 首页 > 面试题库 >

如何在Go中通过反射测试功能集合?

沈淇
2023-03-14
问题内容

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

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 test,只有gorun和问题6511是有没有要求该功能)

为了只编写一个测试,您将 如何使用反射reflect包):

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

但是我想念一个完整的示例来调用函数并在测试中使用返回的值。


问题答案:

一旦我理解了一切都必须使用或返回Value类型,这就是我的想法。
诀窍是使用:

  • ValueOf 为了获得接收者的价值
  • Value.MethodByName寻找那个接收者 价值 的函数 __
  • Value.IsNil测试nil返回值。

测试代码的主要摘录:

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 < 1 {
            t.Error(fname + " should return positive value")
        }
        if err.IsNil() == false {
            t.Error(fname + " shouldn't err")
        }

    }
}

在去操场上看到一个可运行的示例。

请注意,如果使用不存在的函数名称来调用该测试函数,则将出现恐慌。在这里
看到那个例子。

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不支持泛化类型,在Go里面如果你的程序或者数据类型需要操作集合,那么通常是为集合提供一些操作函数。 这里演示了一些操作strings切片的集合函数,你可以使用这些例子来构建你自己的函数。注意在有些情况下,使用内联集合操作代码

  • 问题内容: 我有一个在Go中使用接口定义RPC样式接口的想法。因此,对于给定的服务,我可能会创建一个像这样的接口: 我想做的是使用反射来实现该接口,将方法调用转换为RPC调用,将输入参数编组,然后将结果编组回方法的输出。我知道,如果可以获取输入参数的[] interface {}接口,则可以使用反射进行服务调用。但是,我看不到有任何方法可以使用反射来动态创建一个值,该值可以通过调用使用反射的函数来

  • 在其他一些测试框架中,我习惯于标记测试,例如@really_slow,@front_end

  • 问题内容: 我有一个/ / / etc ..值存储在中,并且想要确定它是否未初始化,这意味着它的值是 要么 我该如何检查? 问题答案: 据我了解,您想要的是: 在谈论接口和时,人们总是对两个截然不同且无关的事物感到困惑: 甲接口值,其是不具有一个接口值 的基础值 。这是接口类型的零值。 非接口值(即,它具有 基础值 ),但是其基础值是其基础类型的零值。例如,基础值是映射,指针或0数字等。 据我了解

  • Google Apps Script支持将事件传递给触发函数的触发器。不幸的是,开发环境将允许您在没有参数传递的情况下测试函数,因此您无法以这种方式模拟事件。如果您尝试,您会得到一个错误,如: 未定义e。 或者 TypeError:无法读取属性*从未定义 (其中未定义) 人们可以将事件视为可选参数,并使用JavaScript中是否有更好的方法来执行可选函数参数的任何技术将默认值插入触发器函数。但这

  • 问题内容: 我有两个字符串: 我通过反思上课 我想要aClass扩展b,例如: 如何实现呢? 如何获得okClass? 谢谢! 问题答案: 除了使用仅通过接口工作的JDK动态代理外,您还可以使用CGLIB或javassist在运行时扩展类。