
我正在尝试使用 go-sqlmock 包测试 sql 查询,其中参数之一是 gosnowflake.array (本质上是切片的包装器)。通常,类似这样的事情需要我创建一个值转换器,我已将其包含在内:
func (opt arrayConverterOption[T]) ConvertValue(v any) (driver.Value, error) {
casted, ok := v.(*[]T)
if ok {
Expect(*casted).Should(HaveLen(len(opt.Expected)))
for i, c := range *casted {
Expect(c).Should(Equal(opt.Expected[i]))
}
} else {
fmt.Printf("Type: %T\n", v)
return v, nil
}
return "TEST_RESULT", nil
}
现在,为提交给查询的每个参数调用此函数。我用它来测试切片中值的正确性,或者如果不是,则传递参数。我遇到的问题是,当我创建 arrayconverteroption[string] 并给它一个 gosnowflake.array(["a", "b", "c"]) 作为参数时,类型断言失败,因为gosnowflake.array 返回一个内部动态类型 *stringarray,它被定义为 *[]string。
所以你可以在这里看到我的困境。一方面,我无法转换 v 因为它是 interface{} 并且我无法为 v 起别名,因为内部类型不是 *[]string,而是 *stringarray。那么,我应该在这里做什么?
正确答案
我没有找到一种方法可以在不引起反思的情况下做到这一点。然而,经过反思,我确实做到了:
var casted []T
var ok bool
value := reflect.ValueOf(v)
if value.Kind() == reflect.Pointer {
if inner := value.Elem(); inner.Kind() == reflect.Slice {
r := inner.Convert(reflect.TypeOf([]T{})).Interface()
casted, ok = r.([]T)
}
}
因此,这段代码专门检查任何指向切片的指针,这就是我的动态类型。然后它使用反射将内部对象转换为我期望的切片类型。之后,我对结果调用 interface() 以从反射值中获取 interface{},然后将其转换为 []t。这样就成功了。如果没有,那么我就不会使用这些动态类型切片之一,并且我可以正常处理该类型。










