golang-Json编码解码

目录

  • 一. 类型映射
  • 二. 输出控制
  • 三. 类型编码
  • 四. 类型解码
  • 五. 输出重写
  • 六. 扩展功能
  • 七. Bson编码

在线工具:https://www.json.cn

一. 类型映射

  • golang和json的数据类型不一致,在编码或解码的过程中必然需要做类型映射。

1.从golang到json:

golang json
bool Boolean
int、float等数字 Number
string String
[]byte(base64编码) String
struct Object,再递归打包
array/slice Array
map Object
interface{} 按实际类型转换
nil null
channel,func UnsupportedTypeError

2.从json到golang:

json golang
Boolean bool
Number float64
String string
Array []interface{}
Object map[string]interface{}
null nil

二. 输出控制

  • 第一个标签为别名,后面控制标签可叠加。
  • 编码或解码的中间数据状态为字节数组
type User struct {
   Name string `json:"user_name"` //修改别名
   Age uint8 `json:",string"` //修改类型
   Addr string `json:"-"` //忽略字段
   Vip bool `json:",omitempty"` //排除缺省值
   Asset int64 `json:"amount,string,omitempty"` //标签叠加
   Skill []string `json:"skill"` //复杂类型
}

func main() {
   obj :=User{Name:"tom",Age:18,Addr:"BJ",Vip:false,Asset:888,Skill:[]string{"golang","python"}}
   bts, _ := json.MarshalIndent(obj,"","\t") //缩进
   fmt.Println(string(bts))
}

三. 类型编码

1.结构体/结构体指针: 映射为Object

func main() {
   obj := struct {A string;B int64}{"a", 1}
   bts, _ := json.Marshal(obj) //结构体
   fmt.Println(string(bts))

   ptr := &obj //结构体指针
   bts, _ = json.Marshal(ptr)
   fmt.Println(string(bts))
}

2.数组/切片: 映射为Array

func main() {
   arr := []string{"a","b","c"}
   bts, _ := json.Marshal(arr)
   fmt.Println(string(bts))
}

3.字典: 映射为Object

func main() {
   mp := map[int]string{1:"a",2:"b",3:"c"}
   bts, _ := json.Marshal(mp)
   fmt.Println(string(bts))
}

四. 类型解码

1.对象解码

func main() {
   str := `{"A":"a","B":1}`
   obj := new(struct {A string;B int64})
   _ = json.Unmarshal([]byte(str), &obj )
   fmt.Println(obj)
}

2.数组解码

func main() {
   arr := `["a","b","c"]`

   slise:=make([]string,0)
   _ = json.Unmarshal([]byte(arr),&slise)
   fmt.Println(slise)
}

3.字典解码

func main() {
   str := `{"1":"a","2":"b","3":"c"}`

   mp := make(map[int]string)
   _ = json.Unmarshal([]byte(str), &mp)
   fmt.Println(mp)
}

五. 输出重写

  • Marshal函数将会递归遍历整个对象,并根据类型数据的MarshalJSON方法打印输出格式。
//自定义的Json时间格式
type Jtime time.Time

//实现了encoding/json/encode.go的Marshaler接口
func (p Jtime) MarshalJSON() ([]byte, error) {
   var stamp = fmt.Sprintf("%d", time.Time(p).Unix())
   //var stamp = fmt.Sprintf("\"%s\"", time.Time(p).Format("2006-01-02 15:04:05"))
   return []byte(stamp), nil
}

func main() {
   obj := struct {
      Name string `json:"nick_name"`
      Date Jtime `json:"create_at"`
   }{"Jack", Jtime(time.Date(2018, 1, 1, 00, 00, 00, 00, time.Local))}

   //编码
   bts, _ := json.MarshalIndent(obj, "", "\t")
   fmt.Println(string(bts))

   //解码
   _ = json.Unmarshal(bts, obj)
   fmt.Printf("%s %q", obj.Name, time.Time(obj.Date))
}

六. 扩展功能

1. Json格式验证:json.Valid()

func main() {
   str := `{"nick_name":"Lucy","user_age":28}` //Object
   str = `["a","b"]` //Array
   ok := json.Valid([]byte(str))
   fmt.Println(ok)
}

2. 自定义编码器:json.NewEncoder()

func main() {
   obj := struct {
      Name string `json:"nick_name"`
      Age uint `json:"user_age"`
   }{"Lucy", 28}

   outer, _ := os.Create("json.txt") //文件输出
   outer = os.Stdout //标准输出
   encoder := json.NewEncoder(outer)
   encoder.SetIndent("", "\t")
   encoder.Encode(obj)
}

3. 缩进处理:json.Indent(),对已编码对字节数组进行缩进

func main() {
   obj := struct {Name string;Age uint}{"Lucy", 28}
   bts, _ := json.Marshal(obj)

   var buf bytes.Buffer
   _=json.Indent(&buf,bts,"","\t")
   buf.WriteTo(os.Stdout)
}

4. Json字符串压缩:json.Compact()

func main() {
   str := `
{
   "nick_name": "Lucy",
   "user_age": 28
}`

   var buf bytes.Buffer
   _ = json.Compact(&buf, []byte(str)) //压缩
   buf.WriteTo(os.Stdout)
}

5. Html编码处理:json.HTMLEscape()

func main() {
   str := `{"content": "<a src=\"http://www.xxx.com\">Link</a>"}`
   var buf bytes.Buffer
   json.HTMLEscape(&buf, []byte(str))
   buf.WriteTo(os.Stdout)
}

6. 二次解码:json.RawMessage,根据json数据包的某个标识,分别解码成不同类型的对象。

type Student struct {
   Sno string //学号
   Name string //姓名
   Major string //专业
}

type Teacher struct {
   Name string //姓名
   Subject string //学科
}

// 任何对象都能装的数据容器,具体类型根据Type字段区分
type JsonObject struct {
   Type int //对象类型
   Obj interface{} //对象数据
}

func main() {
   user := `{"Type":1,"Obj":{"Sno":"S001","Name":"Tom","Major":"computer"}}` //Student
   user = `{"Type":2,"Obj":{"Name":"Bob","Subject":"quantum mechanics"}}` //Teacher

   var obj json.RawMessage
   pkg := JsonObject{Obj: &obj}

   //第一次解码: 先解码外部包装数据对象,获取类型依据
   if err := json.Unmarshal([]byte(user), &pkg); err != nil {
      panic(err)
   }

   //第二次解码: 根据第一次的解码结果,再次对内部对象Obj进行解码
   switch pkg.Type {
   case 1:
      var stu = Student{}
      _ = json.Unmarshal(obj, &stu)
      fmt.Println(stu)
   case 2:
      var tch = Teacher{}
      _ = json.Unmarshal(obj, &tch)
      fmt.Println(tch)
   }
}

七. Bson编码

  • Bson基于json格式,是mongoDB的数据存储格式。
  • 1.速度快:json以字符串形式存储,需要文件扫描,结构匹配。bson是按结构存储,可以精准定位,高效读写。
  • 2.操作简单:json无数据类型,是基于字符的操作,面临很大的操作开销。bson可以指定数据类型。
  • 3.字节数组:二进制的存储不再需要先base64转换后再存成json,大大减少了计算开销和数据大小。
import "gopkg.in/mgo.v2/bson"
type Person struct {
   Name string `bson:"nick_name"`
   Age int32 `bson:"-"`
   Phone string `bson:",omitempty"`
}
func main() {
   p := &Person{"Bob", 18, ""}

   bytes, _ := bson.Marshal(p) //Bson编码
   fmt.Printf("%q\n", bytes)

   um := &Person{}
   bson.Unmarshal(bytes, &um) //strust解码
   fmt.Println(um)

   mp := bson.M{}
   bson.Unmarshal(bytes, mp) //map解码
   fmt.Println(mp)
}

参考:
http://www.cnblogs.com/chuanheng/p/go_bson_struct.html
http://blog.csdn.net/tiaotiaoyly/article/details/38942311
https://studygolang.com/articles/2552
http://labix.org/gobson
http://blog.csdn.net/hengyunabc/article/details/6897540
https://godoc.org/gopkg.in/mgo.v2/bson

原文地址:https://www.cnblogs.com/Hollson/p/11248460.html

时间: 2024-08-01 16:41:37

golang-Json编码解码的相关文章

golang json 编码解码

golang  "encoding/json"包实现了json对象的编解码 一.编码 func Marshal(v interface{}) ([]byte, error) Marshal函数使用下面的基于类型的默认编码格式: 布尔类型编码为json布尔类型. 浮点数.整数和Number类型的值编码为json数字类型. 字符串编码为json字符串.角括号"<"和">"会转义为"\u003c"和"\u003

[转]Golang Gob编码

Golang Gob编码 2012-08-24 09:47 by 轩脉刃, 5119 阅读, 1 评论, 收藏, 编辑 gob是Golang包自带的一个数据结构序列化的编码/解码工具.编码使用Encoder,解码使用Decoder.一种典型的应用场景就是RPC(remote procedure calls). gob和json的pack之类的方法一样,由发送端使用Encoder对数据结构进行编码.在接收端收到消息之后,接收端使用Decoder将序列化的数据变化成本地变量. 有一点需要注意, 发送

网络编程 -- RPC实现原理 -- Netty -- 迭代版本V3 -- 编码解码

网络编程 -- RPC实现原理 -- 目录 啦啦啦 V2--Netty -- pipeline.addLast(io.netty.handler.codec.MessageToMessageCodec<INBOUND_IN, OUTBOUND_IN>) 覆写编码解码方法. pipeline相当于拦截器.在pipeline中添加MessageToMessageCodec接口的实现类,该接口的实现类中的encode()方法自动将发送的Object对象转换为ByteBuf,decode()方法自动将

自定义协议的编码解码

2015.4.1 wqchen. 转载请注明出处 http://www.cnblogs.com/wqchen/p/4385798.html 本文介绍的是一个自定义协议的编码解码工具的实现. 游戏开发中,前端后端协议一般都会协商定制通信协议的格式,统一格式后用程序脚本对应前端和后端的编程语言,分别生成一份协议的编码和解码方案,便于协议的一致性. 这样的工具有很多,比较出名的是google的protobuf,它可以支持很多种编程语言.我也曾试用过protobuf,看过一点它的实现,protobuf完

在JavaScript中使用json.js:使得js数组转为JSON编码

在json的官网中下载json.js,然后在script中引入,以使用json.js提供的两个关键方法. 1.数组对象.toJSONString() 这个方法将返回一个JSON编码格式的字符串,用来表示类型中的数据. 演示: <!DOCTYPE html PUBLIC "-//W3C//DTD XHTML 1.0 Transitional//EN" "http://www.w3.org/TR/xhtml1/DTD/xhtml1-transitional.dtd"

理解netty对protocol buffers的编码解码

一,netty+protocol buffers简要说明 Netty是业界最流行的NIO框架之一优点:1)API使用简单,开发门槛低:2)功能强大,预置了多种编解码功能,支持多种主流协议:3)定制能力强,可以通过ChannelHandler对通信框架进行灵活的扩展:4)性能高,通过与其它业界主流的NIO框架对比,Netty的综合性能最优:5)成熟.稳定,Netty修复了已经发现的所有JDK NIO BUG,业务开发人员不需要再为NIO的BUG而烦恼:6)社区活跃,版本迭代周期短,发现的BUG可以

golang json处理struct未导出成员

我们用golang的json来marshal一个结构体的时候,结构体的未导出的成员将无法被json访问,也就是不会出现json编码的结果里(也就是小写的成员没法导出) 这个是由于技术的上问题引起的:golang的结构体里的成员的名字如果以小写字母开头,那么其他的包是无法访问的,也就是json无法访问我们的结构体里小写字母开头的成员 这个可以有两种方法解决 1. struct的成员用大写开头,然后加tag 2. 实现json.Marshaler接口 第一种方法比较常见这儿就不详细展开了 第二种方法

PHP 汉字 特殊字符 UNICODE 编码 解码 高性能简洁实现方案

网上,针对汉字.特殊字符的UNICODE 编码.解码实现,方法诸多,但是大多是复制粘贴,没有新意! PHP UNICODE 汉字 编码: var_dump(json_encode('2018 ABC 我是中国人!网站:http://my.oschina.net/cart/')); 上述就实现了PHP中对汉字.特殊字符的 UNICODE 编码! 多么easy! 上面会输出: string(96) ""2018 ABC \u6211\u662f\u4e2d\u56fd\u4eba\uff0

iOS &#160;Emoji表情编码/解码

之前做弹幕的时候 遇到的表情编码解码问题 表情编码一般是使用unicode编码 ,编码之后的格式 \ud83d\ude18\ud83d\ude18world\u4e16\u754chaha\ud83d\ude17 //编码 NSString *uniStr = [NSString stringWithUTF8String:[_barrageText.text UTF8String]]; NSData *uniData = [uniStr dataUsingEncoding:NSNonLossyA