1 回答
TA贡献1898条经验 获得超8个赞
在 Go 1.18 中,您不能访问公共字段1,也不能访问类型参数的公共方法2。这些特性之所以不能工作,仅仅是因为它们还没有在该语言中可用。如链接线程所示,常见的解决方案是为接口约束指定方法。
但是,类型*messaging.Message和*messaging.MulticastMessage没有通用的访问器方法,并且在您无法控制的库包中声明。
解决方案1:类型开关
如果联合中的类型较少,则此方法可以正常工作。
func highPriority[T firebaseMessage](message T) T {
switch m := any(message).(type) {
case *messaging.Message:
setConfig(m.Android)
case *messaging.MulticastMessage:
setConfig(m.Android)
}
return message
}
func setConfig(cfg *messaging.AndroidConfig) {
// just assuming the config is always non-nil
*cfg = &messaging.AndroidConfig{}
}
游乐场:https ://go.dev/play/p/9iG0eSep6Qo
解决方案 2:包装方法
这归结为如何在 Go 中向现有类型添加新方法?然后将该方法添加到约束中。如果你有很多结构,它仍然不太理想,但代码生成可能会有所帮助:
type wrappedMessage interface {
*MessageWrapper | *MultiCastMessageWrapper
SetConfig(c foo.Config)
}
type MessageWrapper struct {
messaging.Message
}
func (w *MessageWrapper) SetConfig(cfg messaging.Android) {
*w.Android = cfg
}
// same for MulticastMessageWrapper
func highPriority[T wrappedMessage](message T) T {
// now you can call this common method
message.SetConfig(messaging.Android{"some-value"})
return message
}
游乐场:https ://go.dev/play/p/JUHp9Fu27Yt
方案三:反射
如果你有很多结构,你可能最好使用反射。在这种情况下,类型参数不是严格需要的,但有助于提供额外的类型安全。请注意,结构和字段必须是可寻址的才能起作用。
func highPriority[T firebaseMessage](message T) T {
cfg := &messaging.Android{}
reflect.ValueOf(message).Elem().FieldByName("Android").Set(reflect.ValueOf(cfg))
return message
}
游乐场:https ://go.dev/play/p/3DbIADhiWdO
- 1 回答
- 0 关注
- 90 浏览
添加回答
举报