Services: - Orders: - ID: $save ID1 SupplierOrderCode: $SupplierOrderCode - ID: $save ID2 SupplierOrderCode: 111111
我想将此yaml字符串转换为json,因为源数据是动态的,所以我无法将其映射到结构:
var body interface{} err := yaml.Unmarshal([]byte(s), &body)
然后我想再次将该接口转换为json字符串:
b, _ := json.Marshal(body)
但是发生错误:
panic: json: unsupported type: map[interface {}]interface {}
前言: 我优化并改进了以下解决方案,并将其作为库发布在这里:github.com/icza/dyno。以下convert()功能可作为dyno.ConvertMapI2MapS()。
github.com/icza/dyno
convert()
dyno.ConvertMapI2MapS()
问题在于,如果您使用最通用的interface{}类型来解组,则github.com/go- yaml/yaml包用于解组键- 值对的默认类型将是map[interface{}]interface{}。
interface{}
github.com/go- yaml/yaml
map[interface{}]interface{}
第一个想法是使用map[string]interface{}:
map[string]interface{}
var body map[string]interface{}
但是,如果yaml配置的深度大于一,则此尝试将失败,因为此body映射将包含其他类型为的映射map[interface{}]interface{}。
body
问题在于深度是未知的,并且可能有除地图以外的其他值,因此使用map[string]map[string]interface{}效果不好。
map[string]map[string]interface{}
一种可行的方法是让yaml解组为type的值interface{},然后 递归地 遍历结果,并将遇到的每个值转换map[interface{}]interface{}为一个map[string]interface{}值。地图和切片都必须处理。
yaml
这是此转换器功能的示例:
func convert(i interface{}) interface{} { switch x := i.(type) { case map[interface{}]interface{}: m2 := map[string]interface{}{} for k, v := range x { m2[k.(string)] = convert(v) } return m2 case []interface{}: for i, v := range x { x[i] = convert(v) } } return i }
并使用它:
func main() { fmt.Printf("Input: %s\n", s) var body interface{} if err := yaml.Unmarshal([]byte(s), &body); err != nil { panic(err) } body = convert(body) if b, err := json.Marshal(body); err != nil { panic(err) } else { fmt.Printf("Output: %s\n", b) } } const s = `Services: - Orders: - ID: $save ID1 SupplierOrderCode: $SupplierOrderCode - ID: $save ID2 SupplierOrderCode: 111111 `
输出:
Input: Services: - Orders: - ID: $save ID1 SupplierOrderCode: $SupplierOrderCode - ID: $save ID2 SupplierOrderCode: 111111 Output: {"Services":[{"Orders":[ {"ID":"$save ID1","SupplierOrderCode":"$SupplierOrderCode"}, {"ID":"$save ID2","SupplierOrderCode":111111}]}]}
需要注意的一件事:通过Go映射从yaml转换为JSON,您将失去项目的顺序,因为Go映射中的元素(键-值对)未排序。这可能是问题,也可能不是问题。