我正在嘗試使用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
。這樣就成功了。如果沒有,那麼我就不會使用這些動態類型切片之一,並且我可以正常處理該類型。
以上是從介面轉換私有動態型別{}的詳細內容。更多資訊請關注PHP中文網其他相關文章!