go json数据转发

shanyin · 2019-09-10 00:32:46 · 781 次点击 · 预计阅读时间 2 分钟 · 大约8小时之前 开始浏览    
这是一个创建于 2019-09-10 00:32:46 的文章,其中的信息可能已经有所发展或是发生改变。

案例

例如,有个 GET 接口,可以批量获取用户信息????

> curl 'http://localhost:8080/user/1,2'
[
    {
        "user_id":1,
        "other_suff":...
    },
    {
        "user_id":2,
        "other_suff":...
    }
]

同时,我们要将用户信息和他们的某些订单信息放在一起,组装成为????的接口,满足其他业务需求。

[
    {
        "user_info":{
            "user_id":1,
            "other_suff":...
        },
        "order_info":{
            "order_id":1,
            "user_id":1,
            "other_suff":...
        }
    },
    {
        "user_info":{
            "user_id":2,
            "other_suff":...
        },
        "order_info":{
            "order_id":2,
            "user_id":2,
            "other_suff":...
        }
    }
]

分析

解决这个问题很简单:把user信息和order信息的json用工具解析得到结构体,然后调用他们的接口得到数据,根据id关联和拼装,最后返回。

这样的做法存在的一个问题是,代码解析了user和order的完整结构。如果user接口返回的用户信息增加了字段,我们这里的结构体要同步更新,否则我们给出的数据就是不完整的。(这可能是很痛苦的,你要求别的团队加字段,得排期...)

其实我们作为数据的“中间商”,只关心user接口json里的 user_id ,我们使用这个字段关联order数据。对于user信息里的 other_suff 或者其他数据,我们并不关心,只要保证完整传出去就好了。

根据 https://golang.org/pkg/encodi... ,可以知道直接丢一个 map[string]interface{}json.Unmarshal 也可以正常解析的,于是我们可以写出比较通用的透传代码。

type Content []map[string]interface{}

func (c Content) GetByFieldName(name string, defaultVal interface{}) infterface{} {
    for _, item := range c {
        val, ok := item[name]
        if !ok {
            continue
        }
        if val == nil {
            return defaultVal
        }
        return val
    }
    return defaultVal
}

func getUserContentByIDs(ids []int) Content {
    ...
    var c Content
    err := json.Unmarshal(jsonData, &c) 
    ...
    return c
}

func getOrderContentByUserIDs(ids []int) Content {.../*同上*/}

func Handler(userIDs []int) []Combine {

    users := getUserContentByIDs(userIDs)
    orders := getOrderContentByUserIDs(userIDs)

    // 这里假设用户和订单是一对一的关系
    ret := make([]Combine, 0, len(users))
    for _, u := range users {
        userID := u.GetByFieldName("user_id", 0)
        if userID == 0 {
            continue
        }
        for _, o := range orders {
            orderUserID := o.GetByFieldName("user_id", 0)
            if userID == orderUserID {
                ret = append(ret, Combine{
                    UserInfo:  u,
                    OrderInfo: o,
                })
                break
            }
        }
    }
    return ret
}

更进一步

  • 在上面的例子中,每次查询Content都要遍历数组。如果数据量大或者查询频繁,可以在初始化Content的时候,根据item的唯一标标识,再给Content根据封装一个map,提高查询效率。
  • 从json包的文档可以看到,对于嵌套的结构,用 interface{} 解析以后就是嵌套的 map[string]interface{} , 例如
{
    "foo": "a",
    "bar": {
        "hello": "world",
        "good": "morning"
    }
}

如果用 map[string]interface{} 解析,得到的是map是

map[string]interface{} {
    "foo": "a",
    "bar": map[string]interface{} {
        "hello": "world",
        "good":  "morning"
    }
}

有疑问加站长微信联系(非本文作者)

本文来自:Segmentfault

感谢作者:shanyin

查看原文:go json数据转发

入群交流(和以上内容无关):加入Go大咖交流群,或添加微信:liuxiaoyan-s 备注:入群;或加QQ群:692541889

781 次点击  
加入收藏 微博
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传