【问题标题】:golang fails to parse json for reflection created objectgolang 无法为反射创建的对象解析 json
【发布时间】:2015-11-04 19:31:27
【问题描述】:

我尝试在 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]interface 而不是我的消息。 我做了一个简单的例子来重现这个问题:

http://play.golang.org/p/QEyDN9vztr

【问题讨论】:

  • 在您的示例中,您使用reflect.Zero 而不是reflect.New 作为destination3。更改为New 即可解决问题。但在你的问题中是New,所以我不确定发生了什么。
  • 我用New 重新制作了示例,但结果与play.golang.org/p/Ts0jvApwtY 相同
  • 我没有提到val := reflection.New(type).Interface(); json.Unmarshal(data, val) 有效,但我不需要指针,而是值本身。在这种情况下,我无法进行转换 return *val - 因为错误 invalid indirect

标签: json reflection go unmarshalling


【解决方案1】:

请阅读这篇文章:http://research.swtch.com/interfaces,尤其是“内存优化”。

根据定义,接口{}由两个指针组成 - 指向方法表(例如类型)和它所保存的数据。所以对于

var destination3 interface{} = reflect.Zero(reflect.TypeOf(Message{})).Interface()

它是空的方法表(因为interface{} 没有方法)并引用Message{}。从中获取引用会返回对该结构的引用,因此 unmarhal 会用与 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)
}

在您的情况下,使用 Zero 相当于上面示例中的 m3。使用 New 相当于 m2。

【讨论】:

  • 非常感谢!您的评论说明了问题的核心,这篇文章非常有用 - 不幸的是我之前没有看到它!
  • И спасибо вам по-русски :) Я догадывался, что примерно в эту сторону стоит копать, но до конца не ошёл。 Я на го пишу не так давно и стараюсь вкапываться во все детали, а не писать лишь бы заработало。 А статья, которую вы скинула - действительно очень интересная。 К сожалению далеко не все подобные материалы по go лежат на поверхности - сам бы я вряд ли когда её нашёл...
  • Всегда пожалуйста :) Задавайте вопросы - будем помогать, чем сможем。
【解决方案2】:

我已经找到了做我需要的方法

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() - 我错了吗?

【讨论】:

    猜你喜欢
    • 2016-11-14
    • 1970-01-01
    • 1970-01-01
    • 2017-06-04
    • 2022-11-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多