设为首页 收藏本站
查看: 1000|回复: 0

[经验分享] golang json用法讲解

[复制链接]

尚未签到

发表于 2018-9-20 11:26:16 | 显示全部楼层 |阅读模式
简介

json格式可以算我们日常最常用的序列化格式之一了,Go语言作为一个由Google开发,号称互联网的C语言的语言,自然也对JSON格式支持很好。但是Go语言是个强类型语言,对格式要求极其严格而JSON格式虽然也有类型,但是并不稳定,Go语言在解析来源为非强类型语言时比如PHP等序列化的JSON时,经常遇到一些问题诸如字段类型变化导致无法正常解析的情况,导致服务不稳定。所以本篇的主要目的


  • 就是挖掘Golang解析json的绝大部分能力
  • 比较优雅的解决解析json时存在的各种问题
  • 深入一下Golang解析json的过程



  • Golang解析JSON之Tag篇

  •   一个结构体正常序列化过后是什么样的呢?

  

package main  
import (
  "encoding/json"
  "fmt"
  
)
  

  
// Product 商品信息
  
type Product struct {
  Name      string
  ProductID int64
  Number    int
  Price     float64
  IsOnSale  bool
  
}
  

  
func main() {
  p := &Product{}
  p.Name = "Xiao mi 6"
  p.IsOnSale = true
  p.Number = 10000
  p.Price = 2499.00
  p.ProductID = 1
  data, _ := json.Marshal(p)
  fmt.Println(string(data))
  
}
  

  

  
//结果
  
{"Name":"Xiao mi 6","ProductID":1,"Number":10000,"Price":2499,"IsOnSale":true}
  

  2. 何为Tag,tag就是标签,给结构体的每个字段打上一个标签,标签冒号前是类型,后面是标签名
  

// Product _  
type Product struct {
  Name      string  `json:"name"`
  ProductID int64   `json:"-"` // 表示不进行序列化
  Number    int     `json:"number"`
  Price     float64 `json:"price"`
  IsOnSale  bool    `json:"is_on_sale,string"`
  
}
  

  
// 序列化过后,可以看见
  {"name":"Xiao mi 6","number":10000,"price":2499,"is_on_sale":"false"}
  

  3. omitempty,tag里面加上omitempy,可以在序列化的时候忽略0值或者空值
  

package main  

  
import (
  "encoding/json"
  "fmt"
  
)
  

  
// Product _
  
type Product struct {
  Name      string  `json:"name"`
  ProductID int64   `json:"product_id,omitempty"`
  Number    int     `json:"number"`
  Price     float64 `json:"price"`
  IsOnSale  bool    `json:"is_on_sale,omitempty"`
  
}
  

  
func main() {
  p := &Product{}
  p.Name = "Xiao mi 6"
  p.IsOnSale = false
  p.Number = 10000
  p.Price = 2499.00
  p.ProductID = 0
  

  data, _ := json.Marshal(p)
  fmt.Println(string(data))
  
}
  
// 结果
  
{"name":"Xiao mi 6","number":10000,"price":2499}
  

  4. type,有些时候,我们在序列化或者反序列化的时候,可能结构体类型和需要的类型不一致,这个时候可以指定,支持string,number和boolean
  

package main  

  
import (
  "encoding/json"
  "fmt"
  
)
  

  
// Product _
  
type Product struct {
  Name      string  `json:"name"`
  ProductID int64   `json:"product_id,string"`
  Number    int     `json:"number,string"`
  Price     float64 `json:"price,string"`
  IsOnSale  bool    `json:"is_on_sale,string"`
  
}
  

  
func main() {
  

  var data = `{"name":"Xiao mi 6","product_id":"10","number":"10000","price":"2499","is_on_sale":"true"}`
  p := &Product{}
  err := json.Unmarshal([]byte(data), p)
  fmt.Println(err)
  fmt.Println(*p)
  
}
  
// 结果
  

  
{Xiao mi 6 10 10000 2499 true}
  

  




  • 下面讲一讲Golang如何自定义解析JSON,Golang自带的JSON解析功能非常强悍
说明
  很多时候,我们可能遇到这样的场景,就是远端返回的JSON数据不是你想要的类型,或者你想做额外的操作,比如在解析的过程中进行校验,或者类型转换,那么我们可以这样或者在解析过程中进行数据转换

实例
  

package main  

  
import (
  "bytes"
  "encoding/json"
  "fmt"
  
)
  

  
// Mail _
  
type Mail struct {
  Value string
  
}
  

  
// UnmarshalJSON _
  
func (m *Mail) UnmarshalJSON(data []byte) error {
  // 这里简单演示一下,简单判断即可
  if bytes.Contains(data, []byte("@")) {
  return fmt.Errorf("mail format error")
  }
  m.Value = string(data)
  return nil
  
}
  

  
// UnmarshalJSON _
  
func (m *Mail) MarshalJSON() (data []byte, err error) {
  if m != nil {
  data = []byte(m.Value)
  }
  return
  
}
  

  
// Phone _
  
type Phone struct {
  Value string
  
}
  

  
// UnmarshalJSON _
  
func (p *Phone) UnmarshalJSON(data []byte) error {
  // 这里简单演示一下,简单判断即可
  if len(data) != 11 {
  return fmt.Errorf("phone format error")
  }
  p.Value = string(data)
  return nil
  
}
  

  
// UnmarshalJSON _
  
func (p *Phone) MarshalJSON() (data []byte, err error) {
  if p != nil {
  data = []byte(p.Value)
  }
  return
  
}
  

  
// UserRequest _
  
type UserRequest struct {
  Name  string
  Mail  Mail
  Phone Phone
  
}
  

  
func main() {
  user := UserRequest{}
  user.Name = "ysy"
  user.Mail.Value = "yangshiyu@x.com"
  user.Phone.Value = "18900001111"
  fmt.Println(json.Marshal(user))
  
}
  

  


为什么要这样?
  如果是客户端开发,需要开发大量的API,接收大量的JSON,在开发早期定义各种类型看起来是很大的工作量,不如写 if else 判断数据简单暴力。但是到开发末期,你会发现预先定义的方式能极大的提高你的代码质量,减少代码量。下面实例1和实例2,谁能减少代码一目了然
  

实例1,if else做数据校验  
// UserRequest _
  
type UserRequest struct {
  Name  string
  Mail  string
  Phone string
  
}
  
func AddUser(data []byte) (err error) {
  user := &UserRequest{}
  err = json.Unmarshal(data, user)
  if err != nil {
  return
  }
  //
  if isMail(user.Mail) {
  return fmt.Errorf("mail format error")
  }
  

  if isPhone(user.Phone) {
  return fmt.Errorf("phone format error")
  }
  

  // TODO
  return
  
}
  

  
实例2,利用预先定义好的类型,在解析时就进行判断
  
// UserRequest _
  
type UserRequest struct {
  Name  string
  Mail  Mail
  Phone Phone
  
}
  

  
func AddUser(data []byte) {
  user := &UserRequest{}
  err = json.Unmarshal(data, user)
  if err != nil {
  return
  }
  

  // TODO
  

  
}
  

  

  转自:http://www.cnblogs.com/yangshiyu/p/6942414.html
  https://www.cnblogs.com/52php/p/6518728.html



运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-598808-1-1.html 上篇帖子: Golang教程:Map 下篇帖子: Golang实现断点续传
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表