我删减了一些生产 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: %s\n", es.Name)
}
func (es *BetaStruct) Initialize() {
fmt.Printf("In Beta: %s\n", 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