golang 无法为反射创建的对象解析 json
golang fails to parse json for reflection created object
我尝试在 go 中编写简单的消息协议,但遇到了问题。我有很多消息类型,我想要一个这样的字典来处理消息:
var dict map[reflect.Type]int = map[reflect.Type]int{
reflect.TypeOf(DataMessage{}): 1000,
reflect.TypeOf(TextMessage{}): 1001,
//....
}
func GetMessageTypeId(value interface{}) int {
if id, ok := dict[reflect.TypeOf(value)]; ok {
return id
} else {
return -1
}
}
func GetValueByTypeId(typeId int) interface{} {
for typeDec, id := range dict {
if id == typeId {
return reflect.Zero(typeDec).Interface()
}
}
fmt.Println("Unknown message type", typeId)
return nil
}
它工作正常,但是当我使用 GetValueByTypeId 实例化消息并尝试将 json 解组到其中时 - 我收到 map[string] 接口而不是我的消息。
我做了一个简单的例子来重现这个问题:
我找到了做我需要的方法
val := reflect.New(reflect.TypeOf(Message{}))
json.Unmarshal(data, val.Interface())
return val.Elem().Interface()
http://play.golang.org/p/8g9FSg3MSj
但是第一个版本有问题吗???
看起来 reflect.Zero(type)
应该等同于 reflect.New(type).Elem()
- 我错了吗?
请阅读这篇文章:http://research.swtch.com/interfaces,尤其是"Memory Optimizations"。
根据定义,接口{}由两个指针组成 - 指向方法 table(例如类型)和它保存的数据。所以对于
var destination3 interface{} = reflect.Zero(reflect.TypeOf(Message{})).Interface()
它是空方法 table(因为 interface{}
没有方法)和对 Message{}
的引用。从中引用 returns 对该结构的引用,以便解组用任何匹配项覆盖它 interface{}
.
如果数据 interface{}
变量保存的是一个指针本身,那么它被优化为使用这个指针而不是创建 interface{}
结构。所以引用它给出了对原始变量的引用。
http://play.golang.org/p/KsIS29rUAX
package main
import "fmt"
func main() {
var m1 struct{ Data string }
var m2 interface{}
var m3 interface{}
m2 = &m1
m3 = m1
fmt.Printf("&m1=%p m2=%p &m3=%p\n", &m1, m2, &m3)
}
在您的例子中,使用零相当于上面示例中的 m3。使用 New 相当于 m2.
我尝试在 go 中编写简单的消息协议,但遇到了问题。我有很多消息类型,我想要一个这样的字典来处理消息:
var dict map[reflect.Type]int = map[reflect.Type]int{
reflect.TypeOf(DataMessage{}): 1000,
reflect.TypeOf(TextMessage{}): 1001,
//....
}
func GetMessageTypeId(value interface{}) int {
if id, ok := dict[reflect.TypeOf(value)]; ok {
return id
} else {
return -1
}
}
func GetValueByTypeId(typeId int) interface{} {
for typeDec, id := range dict {
if id == typeId {
return reflect.Zero(typeDec).Interface()
}
}
fmt.Println("Unknown message type", typeId)
return nil
}
它工作正常,但是当我使用 GetValueByTypeId 实例化消息并尝试将 json 解组到其中时 - 我收到 map[string] 接口而不是我的消息。 我做了一个简单的例子来重现这个问题:
我找到了做我需要的方法
val := reflect.New(reflect.TypeOf(Message{}))
json.Unmarshal(data, val.Interface())
return val.Elem().Interface()
http://play.golang.org/p/8g9FSg3MSj
但是第一个版本有问题吗???
看起来 reflect.Zero(type)
应该等同于 reflect.New(type).Elem()
- 我错了吗?
请阅读这篇文章:http://research.swtch.com/interfaces,尤其是"Memory Optimizations"。
根据定义,接口{}由两个指针组成 - 指向方法 table(例如类型)和它保存的数据。所以对于
var destination3 interface{} = reflect.Zero(reflect.TypeOf(Message{})).Interface()
它是空方法 table(因为 interface{}
没有方法)和对 Message{}
的引用。从中引用 returns 对该结构的引用,以便解组用任何匹配项覆盖它 interface{}
.
如果数据 interface{}
变量保存的是一个指针本身,那么它被优化为使用这个指针而不是创建 interface{}
结构。所以引用它给出了对原始变量的引用。
http://play.golang.org/p/KsIS29rUAX
package main
import "fmt"
func main() {
var m1 struct{ Data string }
var m2 interface{}
var m3 interface{}
m2 = &m1
m3 = m1
fmt.Printf("&m1=%p m2=%p &m3=%p\n", &m1, m2, &m3)
}
在您的例子中,使用零相当于上面示例中的 m3。使用 New 相当于 m2.