解决golang json解析出现值为空的问题

 更新时间:2020年12月24日 11:26:44   作者:一只APE的进化史  
这篇文章主要介绍了解决golang json解析出现值为空的问题,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧

我是通过beego框架,将请求过来的json进行解析,并将值保存在结构体中

--------------------1---------------------
 request := UpdateCommentRequestData{}
 req := common.Request{Data: request}
 err := json.Unmarshal(controller.Ctx.Input.RequestBody, &req)
------------------------------------------

其中 UpdateCommentRequestData的结构是这样的

type UpdateCommentRequestData struct {
 Id []string `json:"id"`
}

common.request的结构是这样的

type Request struct {
 UserId uint64 `json:"userId,string"`
 Data interface{} `json:"data"`
}

我使用1中的代码进行解析,发现request.Id的值是空的,但是传来的json是存在Id值的,当时一头雾水,就不断在日志中打印,后来定位到是数据类型存在问题,

在1中的代码里面,Data字段传的是request的值,是值的拷贝,也就是说,json解析后的数据并不是赋值到reques中,所以使用request.Id并不会取到值,

如果将代码改成这样,再使用request.Id就可以取到值了

req := common.Request{Data: request},

补充:golang Unmarshal拿不全数据问题

说明:这个问题出现在后端调用json.Unmarshal方法去解析数据库中存的数据时,解析出来的结果中只能拿到部分数据,json格式经检查后正确无误,同时也没有字段名出错等低级错误。

首先来看要解析后的go结构体

type ParamConfig struct {
 //标识Id
 Id  string
 //抓拍目标参数配置
 SnapObjConfig *SnapObjConfig
 //默认去重参数配置
 DefaltDeweightConfig *DefaltDeweightConfig
}
//抓拍目标参数结构
type SnapObjConfig struct {
 //分辨率参数
 Distinguish *Distinguish
 //机动车配置
 vehicle *DataConfig
 //非机动车配置
 nonmotor *DataConfig
 //行人配置
 pedestrian *DataConfig
 //人脸配置
 face *DataConfig
}
//分辨率结构
type Distinguish struct {
 //分辨率值
 DistinguishRate int32
}
 
type DataConfig struct {
 //最小宽度
 MinWeight int32
 //最小高度
 MinHight int32
}
//默认去重参数结构
type DefaltDeweightConfig struct {
 vehicle *DeweightNum
 nonmotor *DeweightNum
 pedestrian *DeweightNum
 face *DeweightNum
}
//默认参数值结构
type DeweightNum struct {
 Number float32
}

先向数据库中插入一条需要解析的数据

SQL语句如下所示:

INSERT INTO "public"."sys_config"("config_key", "config_value") VALUES ('param_config', '[{"Id":"8149aa8e-1466-469b-ac5e-b0ea72f96129","SnapObjConfig":{"Distinguish":{"DistinguishRate":270},"vehicle":{"MinWeight":128,"MinHight":128},"nonmotor":{"MinWeight":32,"MinHight":64},"pedestrian":{"MinWeight":32,"MinHight":64},"face":{"MinWeight":40,"MinHight":40}},"DefaltDeweightConfig":{"vehicle":{"Number":0.95},"nonmotor":{"Number":0.95},"pedestrian":{"Number":0.95},"face":{"Number":0.95}}}]');

为了方便说明下面在代码中打上详细的log,大码如下:

func (this *CommonController)GetParamConfig(c *gin.Context) {
 searchResp := &models.SearchResp{
 Code: models.ApiStatus_SUCCESS,
 Msg: "successs",
 }
 retParamConfig := make([]*ParamConfig, 0)
 if configs, err := db_model.SysConfigsByConfigKey(this.DB, ParamConfigKey); err != nil && !models.IsEmptyResults(err){
 glog.Infoln(err)
 searchResp.Code = models.ApiStatus_ERROR
 searchResp.Msg = "fail"
 c.JSON(http.StatusInternalServerError, searchResp)
 return
 } else if len(configs) > 0 {
 glog.Infoln("data----------------", configs[0].ConfigValue)
 if err := json.Unmarshal([]byte(configs[0].ConfigValue), &retParamConfig); err != nil {
 glog.Errorln(err)
 searchResp.Code = models.ApiStatus_ERROR
 searchResp.Msg = err.Error()
 c.JSON(http.StatusInternalServerError, searchResp)
 return
 }
 }
 searchResp.Data = retParamConfig
 glog.Infoln("retParamConfig[0].SnapObjConfig.Vehicle----------", retParamConfig[0].SnapObjConfig.vehicle)
 glog.Infoln("retParamConfig[0].SnapObjConfig.nonmotor-----------", retParamConfig[0].SnapObjConfig.nonmotor)
 glog.Infoln("retParamConfig[0].SnapObjConfig.pedestrian------------", retParamConfig[0].SnapObjConfig.pedestrian)
 glog.Infoln("retParamConfig[0].SnapObjConfig.Fsce------------------", retParamConfig[0].SnapObjConfig.face)
 glog.Infoln("retParamConfig[0].DefaltDeweightConfig.Fsce------------------", retParamConfig[0].DefaltDeweightConfig.face)
 glog.Infoln("retParamConfig[0].DefaltDeweightConfig.Fsce------------------", retParamConfig[0].DefaltDeweightConfig.vehicle)
 glog.Infoln("retParamConfig[0].DefaltDeweightConfig.Fsce------------------", retParamConfig[0].DefaltDeweightConfig.nonmotor)
 glog.Infoln("retParamConfig[0].DefaltDeweightConfig.Fsce------------------", retParamConfig[0].DefaltDeweightConfig.pedestrian)
 
 c.JSON(http.StatusOK, searchResp)
}

运行之后如图所示:

很明显从一开始我们就向数据库中存入了数据,同时从日志中可以看出,data中存的是去数据库中获取的数据,数据和刚开始存入到数据库中的值一样,但是调用unmarshal之后却获取不到全部的数据,可以看一下使用postman调用接口之后的返回结果如下:

接口的返回值中只是返回了部分数据,到底是出了什么问题呢?之后我曾仔细的核对完每一个结构字段和数据库中字段的类型,确保并不是这些原因导致的,想了很久不知道这个问题到底是如何发生的,无意中将结构体中的字段名的首字母都变成了大写,经过编译运行之后终于拿到了全部的数据,

有了这个结果之后,我又去仔细的google了一下这个问题,原来结构体中的每一项如果是导出项的时候首字母必须是大写的,但是问题是SQL语句中在数据库中存入的信息都是首字母小写的,检索出来的结果却是大写的,很明显这个处理过程中大小写匹配的问题被忽略掉了,因此要想按照我们的信息随意匹配的话就得在结构体后面加tag,这样解析时就会只匹配tag中的名字,但是tag中的结果不能为空格否则依旧会报错。

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。如有错误或未考虑完全的地方,望不吝赐教。

相关文章

  • Go语言实现websocket推送程序

    Go语言实现websocket推送程序

    这篇文章主要介绍了Go语言实现websocket推送程序,WebSocket是基于TCP的一个双向传输数据的协议,和HTTP协议一样,是在应用层的,他的出现,是为了解决网页进行持久双向传输数据的问题
    2023-01-01
  • go语言计算两个时间的时间差方法

    go语言计算两个时间的时间差方法

    这篇文章主要介绍了go语言计算两个时间的时间差方法,涉及Python操作时间的技巧,需要的朋友可以参考下
    2015-03-03
  • 详解Go语言中配置文件使用与日志配置

    详解Go语言中配置文件使用与日志配置

    这篇文章主要为大家详细讲解一下Go语言中调整项目目录结构、增加配置文件使用和增加日志配置的方法,文中示例代码讲解详细,需要的可以参考一下
    2022-06-06
  • Golang简介与基本语法的学习

    Golang简介与基本语法的学习

    这篇文章主要介绍了Golang简介与基本语法的学习,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2023-04-04
  • golang 输出重定向:fmt Log,子进程Log,第三方库logrus的详解

    golang 输出重定向:fmt Log,子进程Log,第三方库logrus的详解

    这篇文章主要介绍了golang 输出重定向:fmt Log,子进程Log,第三方库logrus的详解,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2020-12-12
  • golang实现京东支付v2版本的示例代码

    golang实现京东支付v2版本的示例代码

    这篇文章主要介绍了golang实现京东支付v2版本,本文给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2021-03-03
  • 利用Golang生成整数随机数方法示例

    利用Golang生成整数随机数方法示例

    这篇文章主要介绍了利用Golang生成整数随机数的相关资料,文中给出了详细的介绍和完整的示例代码,相信对大家具有一定的参考价值,需要的朋友们下面来一起看看吧。
    2017-04-04
  • 初探GO中unsafe包的使用

    初探GO中unsafe包的使用

    unsafe是Go语言标准库中的一个包,提供了一些不安全的编程操作,本文将深入探讨Go语言中的unsafe包,介绍它的使用方法和注意事项,感兴趣的可以了解下
    2023-08-08
  • go实现Redis读写分离示例详解

    go实现Redis读写分离示例详解

    本篇文章将介绍Redis通信协议RESP, 而后在使用go来编写一个中间件,从而来完成Redis读写分离,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-08-08
  • vscode上搭建go开发环境详细完整过程

    vscode上搭建go开发环境详细完整过程

    这篇文章主要给大家介绍了关于vscode上搭建go开发环境的详细完整过程,Go语言或将成为新的主力开发语言,Go是google开发的一种静态强类型、编译型、并发型,并具有垃圾回收功能的编程语言,所以我们有必要学习并掌握它,需要的朋友可以参考下
    2023-10-10

最新评论