给出以下代码:(转载于此处play.golang.org.)
package main
import (
"encoding/json"
"fmt"
)
type User struct {
Id int `json:"id"`
Name string `json:"name"`
}
type Session struct {
Id int `json:"id"`
UserId int `json:"userId"`
}
type Anything interface{}
type Hateoas struct {
Anything
Links map[string]string `json:"_links"`
}
func MarshalHateoas(subject interface{}) ([]byte, error) {
h := &Hateoas{subject, make(map[string]string)}
switch s := subject.(type) {
case *User:
h.Links["self"] = fmt.Sprintf("http://user/%d", s.Id)
case *Session:
h.Links["self"] = fmt.Sprintf("http://session/%d", s.Id)
}
return json.MarshalIndent(h, "", " ")
}
func main() {
u := &User{123, "James Dean"}
s := &Session{456, 123}
json, err := MarshalHateoas(u)
if err != nil {
panic(err)
} else {
fmt.Println("User JSON:")
fmt.Println(string(json))
}
json, err = MarshalHateoas(s)
if err != nil {
panic(err)
} else {
fmt.Println("Session JSON:")
fmt.Println(string(json))
}
}
我试图让呈现的 JSON 看起来correct就我而言,这意味着:
User JSON:
{
"id": 123,
"name": "James Dean",
"_links": {
"self": "http://user/123"
}
}
Session JSON:
{
"id": 456,
"userId": 123,
"_links": {
"self": "http://session/456"
}
}
不幸的是,Go 将匿名成员视为真实命名的事物,因此它采用定义的类型(Anything
) 并如此命名 JSON:
User JSON:
{
"Anything": {
"id": 123,
"name": "James Dean"
},
"_links": {
"self": "http://user/123"
}
}
Session JSON:
{
"Anything": {
"id": 456,
"userId": 123
},
"_links": {
"self": "http://session/456"
}
}
没有关于 JSON 中匿名成员处理的明确文档,来自the docs:
匿名结构体字段通常被编组,就好像它们的内部导出字段是外部结构体中的字段一样,遵循下一段中描述的修改后的常见 Go 可见性规则。具有在其 JSON 标记中给出的名称的匿名结构字段被视为具有该名称,而不是匿名的。
处理匿名结构体字段是 Go 1.1 中的新功能。在 Go 1.1 之前,匿名结构体字段被忽略。要强制忽略当前版本和早期版本中的匿名结构字段,请为该字段指定 JSON 标记“-”。
这并不清楚是否有办法进行平坦化,或者向编组者暗示我正在尝试做什么。
我确信,由于存在特殊情况,可能存在具有特殊含义的魔术名称,可以在 XML 编组器中重命名 XML 文档的根元素。
在这种情况下,我也没有以任何方式附加到代码,我的用例是有一个接受的函数interface{}, *http.Request, http.ResponseWriter
并通过线路写回 HATEOAS 文档,切换传递的类型,以推断哪些链接写回 JSON。 (从而访问请求、请求主机、端口、方案等,以及类型本身以推断 URL 和已知字段等)