我删减了一些生产 GoLang 代码,以创建以下示例:
package main import ( "encoding/json" "fmt" ) func main() { var aData []AlphaStruct var bData []BetaStruct readData(&aData) readData(&bData) // Do other stuff with the loaded structures. } func readData(structSlicePointer any) { json.Unmarshal([]byte( "["+ "{"name": "John Smith"},"+ "{"name": "Jane Doe"}"+ "]"), structSlicePointer) var initializers []Initializer // At this point I want to do: // initializers = structSlicePointer // However I can't find any type declarations or casts that will work with the rest of the code. for _, initializer := range initializers { initializer.Initialize() } } type Initializer interface { Initialize() } type AlphaStruct struct { Name string `json:"name"` } type BetaStruct struct { Name string `json:"name"` } func (es *AlphaStruct) Initialize() { fmt.Printf("In Alpha: %sn", es.Name) } func (es *BetaStruct) Initialize() { fmt.Printf("In Beta: %sn", es.Name) }
我有许多结构,它们都实现相同的接口。
这个想法是,在反序列化它们(从 JSON)之后,我想在每个加载的对象上调用一个方法,以完成对象初始化。一些 JSON 文件具有顶级数组 - 即一旦反序列化,它们就是结构的列表/切片。
但是,我无法找到任何可以让我同时执行以下操作的声明组合:
json.Unmarshal()
显然,我可以将其拆分为两个函数(loadData 和initializeData)来执行这两个函数。
有什么方法可以对其进行重组,使其作为单一方法工作吗?
您可以将该方法声明为通用方法,该方法接受一个带有实现 Initializer
接口的成员结构的切片。
为此,您可以像这样声明您的方法:
func readData[T Initializer](structSlicePointer *[]T) { json.Unmarshal( []byte("[{"name": "John Smith"},{"name": "Jane Doe"}]"), structSlicePointer, ) for _, initializer := range *structSlicePointer { initializer.Initialize() } }
请注意,这将要求您将切片值声明为指针,以便与结构上的指针接收器方法一起使用(例如 []*AlphaStruct
与 []AlphaStruct
):
func main() { var aData []*AlphaStruct var bData []*BetaStruct readData(&aData) readData(&bData) }
输出:
In Alpha: John Smith In Alpha: Jane Doe In Beta: John Smith In Beta: Jane Doe