1

是否可以忽略MarshalJSON结构的自定义实现,而只使用标准编组函数?

该结构很复杂,并且有很多嵌套结构,所有这些都使用 custom MarshalJSON,我想忽略它们。

我觉得应该是微不足道的。你有想法吗?

一些细节

创建新类型的明显解决方案效果不佳,因为嵌套结构仍然使用它们MarshalJSON的 s.

以下是代码示例:

func (de DeploymentExtended) MarshalJSON() ([]byte, error) {
    objectMap := make(map[string]interface{})
    if de.Location != nil {
        objectMap["location"] = de.Location
    }
    if de.Properties != nil {
        objectMap["properties"] = de.Properties
    }
    if de.Tags != nil {
        objectMap["tags"] = de.Tags
    }
    return json.Marshal(objectMap)
}

(来源:https ://github.com/Azure/azure-sdk-for-go/blob/v62.0.0/services/resources/mgmt/2020-10-01/resources/models.go#L366 )

并且有很多属性(例如Name等),我希望在我的 JSON 中看到这些属性(对于Properties其他嵌套结构也是如此)。

此代码的 Python 实现提供了该数据,我的软件使用它,并且我(将代码移植到 Go)也希望能够从我的 Go 程序中导出这些数据。

4

1 回答 1

2

您可以通过以下两种方式执行此操作:

  • 自定义类型(隐藏MarshalJSON方法);或者
  • 自定义封送拆收器(用于在运行时reflect忽略任何MarshalJSON方法)

自定义类型

例如,采用这些嵌套类型:

type Y struct {
    FieldZ string
}
type X struct {
    Name string
    Y    Y
}


func (x *X) MarshalJSON() ([]byte, error) { return []byte(`"DONT WANT THIS"`), nil }
func (y *Y) MarshalJSON() ([]byte, error) { return []byte(`"DEFINITELY DONT WANT THIS"`), nil }

需要隐藏这些类型以避免MarshalJSON调用不需要的方法:

type shadowY struct {
    FieldZ string
}
type shadowX struct {
    Name string
    Y    shadowY
}

//
// transform original 'x' to use our shadow types
//
x2 := shadowX{
    Name: x.Name,
    Y:    shadowY(x.Y),
}

https://go.dev/play/p/vzKtb0gZZov


反射

这是一个reflect基于简单的 JSON 封送器来实现您想要的。它假定所有自定义编组器都使用指针接收器 - 并取消引用指针,因此标准库json.Marshal不会“看到”它们:

func MyJSONMarshal(v interface{}) (bs []byte, err error) {
    k := reflect.TypeOf(v).Kind() // ptr or not?

    if k != reflect.Ptr {
        return json.Marshal(v)
    }

    // dereference pointer
    v2 := reflect.ValueOf(v).Elem().Interface()
    return MyJSONMarshal(v2)
}

YMMV用这种方法。

https://go.dev/play/p/v9YjYRno7RV

于 2022-03-02T16:40:12.237 回答