文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Golang中JSON遇到的坑如何解决

2023-07-05 06:11

关注

本篇内容主要讲解“Golang中JSON遇到的坑如何解决”,感兴趣的朋友不妨来看看。本文介绍的方法操作简单快捷,实用性强。下面就让小编来带大家学习“Golang中JSON遇到的坑如何解决”吧!

空指针会被解析成字符串"null"

type Person struct {Name stringAge  int}func main() {var p *Personbytes, err := json.Marshal(p)checkError(err)fmt.Printf("len:%d, result:%s\n", len(bytes), string(bytes))  // len:4, result:null}func checkError(err error) {if err != nil {fmt.Printf("err:%+v\n", err)}}

json.Marshal一个空指针的时候,得到的结果居然是"null"字符串,我以为是""或者报错。

还有个奇怪的坑

type Person struct {Name stringAge  int}func main() {var p *Persons := `null`err := json.Unmarshal([]byte(s), &p)checkError(err)fmt.Printf("p:%+v\n", p) // p:<nil>}

这个居然不报错,而是得到空指针p

如果把s随便换成其他字符串s := "abc",则报错:invalid character 'a' looking for beginning of value,之前我理解的是null对go来说应该跟abc没有差别,都是字符串。没想到他们是不一样的,下面来深究一下json.UnMarshal底层代码。

在UnMarshal之前它有个checkValid函数

func checkValid(data []byte, scan *scanner) error {scan.reset()for _, c := range data {scan.bytes++if scan.step(scan, c) == scanError {return scan.err}}if scan.eof() == scanError {return scan.err}return nil}

checkValid函数会check每一个字符,调用step函数,step初始值是stateBeginValue

// stateBeginValue is the state at the beginning of the input.func stateBeginValue(s *scanner, c byte) int {if isSpace(c) {return scanSkipSpace}switch c {case '{':s.step = stateBeginStringOrEmptyreturn s.pushParseState(c, parseObjectKey, scanBeginObject)case '[':s.step = stateBeginValueOrEmptyreturn s.pushParseState(c, parseArrayValue, scanBeginArray)case '"':s.step = stateInStringreturn scanBeginLiteralcase '-':s.step = stateNegreturn scanBeginLiteralcase '0': // beginning of 0.123s.step = state0return scanBeginLiteralcase 't': // beginning of trues.step = stateTreturn scanBeginLiteralcase 'f': // beginning of falses.step = stateFreturn scanBeginLiteralcase 'n': // beginning of nulls.step = stateNreturn scanBeginLiteral}if '1' <= c && c <= '9' { // beginning of 1234.5s.step = state1return scanBeginLiteral}return s.error(c, "looking for beginning of value")}

有这么一段代码,这是处理第一个字符的,发现它对第一个字符是n有特殊处理并且设置下一个字符处理函数为stateN

// stateN is the state after reading `n`.func stateN(s *scanner, c byte) int {if c == 'u' {s.step = stateNureturn scanContinue}return s.error(c, "in literal null (expecting 'u')")}

也就是下一个字符必须是u,再下一个字符处理函数为stateNu

// stateNu is the state after reading `nu`.func stateNu(s *scanner, c byte) int {if c == 'l' {s.step = stateNulreturn scanContinue}return s.error(c, "in literal null (expecting 'l')")}

也就是下一个字符必须是l,再下一个字符处理函数为stateNul

// stateNul is the state after reading `nul`.func stateNul(s *scanner, c byte) int {if c == 'l' {s.step = stateEndValuereturn scanContinue}return s.error(c, "in literal null (expecting 'l')")}

也就是下一个字符必须是l,再下一个字符处理函数为stateEndValue。

可见checkValid函数对true,false等都有特殊处理。使用时需要注意。

对于json.Marshal函数,通过调试发现它对空指针也有特殊处理

type ptrEncoder struct {elemEnc encoderFunc}func (pe ptrEncoder) encode(e *encodeState, v reflect.Value, opts encOpts) {if v.IsNil() {e.WriteString("null")return}if e.ptrLevel++; e.ptrLevel > startDetectingCyclesAfter {// We're a large number of nested ptrEncoder.encode calls deep;// start checking if we've run into a pointer cycle.ptr := v.Interface()if _, ok := e.ptrSeen[ptr]; ok {e.error(&UnsupportedValueError{v, fmt.Sprintf("encountered a cycle via %s", v.Type())})}e.ptrSeen[ptr] = struct{}{}defer delete(e.ptrSeen, ptr)}pe.elemEnc(e, v.Elem(), opts)e.ptrLevel--}

如果是空指针则返回字符串"null",并且不会报错。

int类型会被解析成float64

type Person struct {Name stringAge  int}func main() {p := &Person{Name: "text",Age:  18,}bytes, err := json.Marshal(p)checkError(err)pMap := make(map[string]interface{})err = json.Unmarshal(bytes, &pMap)checkError(err)for k, v := range pMap {fmt.Printf("k:%s,v:%+v, vtype:%v\n", k, v, reflect.TypeOf(v))}}func checkError(err error) {if err != nil {fmt.Printf("err:%+v\n", err)}}

结果

k:Name,v:text, vtype:string
k:Age,v:18, vtype:float64

显然,Age类型变成了float64。会造成什么问题呢?当int大小超过6位的时候就变成了科学计数法 比如Age=1234567, 结果为

k:Name,v:text, vtype:string
k:Age,v:1.234567e+06, vtype:float64

这个时候如果直接将map更新到db,原本是int类型的字段变成了float类型,就报错了

到此,相信大家对“Golang中JSON遇到的坑如何解决”有了更深的了解,不妨来实际操作一番吧!这里是编程网网站,更多相关内容可以进入相关频道进行查询,关注我们,继续学习!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯