Go 结构体序列化的实现

目录
  • 更改JSON对象中的键
  • 在JSON对象中隐藏结构体字段
  • 附加内容
    • 结构体标签string指令

本文,我们将回到之前写的showMovieHandler方法,并更新它以返回一个JSON响应,表示系统中的单个电影信息。类似于:

{
    "id": 123,
    "title": "Casablanca", 
    "runtime": 102, 
    "genres": [
        "drama", 
        "romance", 
        "war"
    ],
    "version": 1 
}

我们不使用map序列化来创建这个JSON对象(就像我们在上一节中所做的那样),这次我们将编码一个自定义的Movie结构体。

首先,需要定义一个Movie结构体。我们将在一个新internal/data包中完成此操作,该包稍后将扩展用来封装项目中所有自定义数据类型以及与数据库交互的逻辑。

如果您按照文章步骤操作,请创建一个新的internal/data目录,其中包含一个movies.go文件:

$ mkdir internal/data
$ touch internal/data/movies.go

在这个新文件中,定义Movie结构,像这样:

File: internal/data/movies.go

package main

import (
    "time"
)

type Movie struct {
    ID             int64      //唯一整数ID
    CreatedAt      time.Time  //创建电影到数据库的时间
    Title          string     //电影标题
    Year           int32      //电影发布年份
    Runtime        int32      //电影时长
    Genres         []string   //电影类型(爱情片、喜剧片等)
    Version        int32      //版本号从1开始,每更新一次递增
}

这里需要指出的是,Movie结构体中的所有字段都是可导出的(即以大写字母开头),这对于Go的encoding/json包可见是必要的。在将结构体编码为JSON时,不会包含任何未导出的字段。

现在结构体已经定义完成,让我们更新showMovieHandler处理程序来初始化一个Movie结构体实例,然后使用writeJSON()帮助函数将其作为JSON响应发送给客户端。

实现很简单:

File: cmd/api/movies.go

package main

import (
    "fmt"
    "net/http"
    "time"

    "greenlight.alexedwards.net/internal/data"
)

func (app *application) showMovieHandler(w http.ResponseWriter, r *http.Request) {
    id, err := app.readIDParam(r)
    if err != nil {
        http.NotFound(w, r)
        return
    }

    //创建一个Move结构体实例,包含从请求URL中解析的ID虚构的数据。注意这里故意没有设置Year字段
    movie := date.Movie{
        ID: id,
        CreateAt: time.now(),
        Title: "Casablanca",
        Runtime: 102,
        Genres: []string{"drama", "romance", "war"},
        Version: 1,
    }

    //将结构体序列化为JSON并以HTTP响应发送给客户端
    err = app.writeJSON(w, http.StatusOK, movie, nil)
    if err != nil {
         app.logger.Println(err)
         http.Error(w, "The server encountered a problem and could not process your request", http.StatusInternalServerError)
    }
}

ok,下面试试!

重启API,然后在浏览器中访问localhost:4000/v1/movies/123。你应该会看到一个类似这样的JSON响应:

在这个返回结果中,有几件有趣的事情需要指出:

  • Movie结构体被编码成一个JSON对象,字段名和值作为键/值对。
  • 默认情况下,JSON对象中的键等于结构体中的字段名(ID、CreatedAt、Title等等)。我们稍后将讨论如何自定义JSON键。
  • 如果结构体实例字段没有显式赋值,那么字段零值将序列化为json值。可以在上面的响应中看到——我们没有在Go代码中为Year字段设置值,但它仍然以0值出现在JSON输出中。

更改JSON对象中的键

在Go中序列化结构体的一个好处是,您可以通过使用struct标签注释字段来定制JSON。

struct标签最常见的用途可能是更改JSON对象中出现的键名称。当你的结构体字段名不适合面向公众展示,或者你想在JSON输出中使用另一种大小写样式时,这是很有用的。

为了说明如何实现,对Movies结构体字段打标签,使用蛇形格式:

File: internal/data/movies.go

//使用标记对Movie结构进行注释,以控制json编码的key显示方式。
type Movie struct {
    ID       int64     `json:"id"`
    CreateAt time.Time `json:"created_at"`
    Title    string    `json:"title"`
    Year     int32     `json:"year"`
    Runtime  int32     `json:"runtime"`
    Genres   []string  `json:"genres"`
    Version  int32     `json:"version"`
}

如果你重启服务器并再次访问localhost:4000/v1/movies/123,应该会看到一个类似于这样的带有蛇形键的响应:

在JSON对象中隐藏结构体字段

在定义结构体时候,通过使用omitempty可以控制对应字段在JSON中的可见性。当您不希望JSON输出中出现特定的结构体字段时,可以使用-(连字符)指令。这对包含和用户不相关的内部系统信息的字段或不想公开的敏感信息(如密码哈希值)非常有用。

相反,当且仅当struct字段值为空时,omitempty指令会在JSON输出中隐藏字段,其中empty被定义为:

  • 等于false,0或“”
  • 空数组,切片或map
  • nil指针或接口值为nil

为了演示如何使用这些指令,我们对Movie结构进行更多的改造。CreatedAt字段与我们的最终用户无关,所以我们使用-指令在输出中将其隐藏。我们还将使用omitempty指令在输出中隐藏Year、Runtime和types字段,当且仅当它们为空时生效。

继续并像下面这样更新struct标签:

File:interface/data/movies.go

package data

....

type Movie struct {
    ID       int64     `json:"id"`
    CreateAt time.Time `json:"-"`       //使用-指令
    Title    string    `json:"title"`
    Year     int32     `json:"year,omitempty"`            //添加omitempty
    Runtime  int32     `json:"runtime,omitempty"`         //添加omitempty
    Genres   []string  `json:"genres,omitempty"`          //添加omitempty
    Version  int32     `json:"version"`
}

如果你想使用omitempty而不改变键名,那么你可以在struct标签中保留它为空-如:json:",omitempty"。注意,逗号是必要的。

现在,当你重新启动应用程序并刷新你的web浏览器时,你应该会看到如下响应:

我们可以在这里看到,CreatedAt结构字段不再出现在JSON中,而且Year字段(值为0)也没有出现,这要感谢omitempty指令。其他字段使用了omitempty不受影响(例如Runtime和Genres)。

注意:您还可以通过简单地将结构体字段设置为不可导出来防止它出现在JSON序列化中。但使用json:“-“标记通常是一个更好的选择:明确告知阅读代码的人,你不希望该字段包含在json。

旧版本的go vet如果你试图在未导出的字段上使用struct标记会引发错误,但现在在go 1.16中已经修复了这个问题。

附加内容

结构体标签string指令

最后一个不太常用的struct标记指令是string。可以使用这个标签明确表示字段值序列化成JSON字符串类型。例如,如果我们希望Runtime字段的值表示为一个JSON字符串 (而不是数字)我们可以像这样使用string指令:

type Movie struct {
    ID       int64     `json:"id"`
    CreateAt time.Time `json:"-"`       //使用-指令
    Title    string    `json:"title"`
    Year     int32     `json:"year,omitempty"`   
    Runtime  Runtime   `json:"runtime,omitempty,string"` 
    Genres   []string  `json:"genres,omitempty"`       
    Version  int32     `json:"version"`
}

JSON序列化结果如下所示:

{
"id": 123,
"title": "Casablanca",
"runtime": "102",   //这是字符串
"genres": [
    "drama", 
    "romance", 
    "war"
    ],
"version": 1 
}

注意string指令只对int, uint, float*或bool类型的字段有效。对于任何其他类型的结构体字段没有作用。

到此这篇关于Go 结构体序列化的实现的文章就介绍到这了,更多相关Go 结构体序列化内容请搜索我们以前的文章或继续浏览下面的相关文章希望大家以后多多支持我们!

(0)

相关推荐

  • golang实现java uuid的序列化方法

    目前只实现了java生成的固定的uuid:85bb94b8-fd4b-4e1c-8f49-3cedd49d8f28的序列化 package main import ( "encoding/binary" "encoding/json" "fmt" "log" "os" "strings" "time" "github.com/Shopify/sarama&q

  • golang:json 反序列化的[]和nil操作

    我就废话不多说了,大家还是直接看代码吧~ package main import ( "encoding/json" "log" ) type JS struct { A []string } func main() { v1 := &JS{} v2 := &JS{A: []string{}} o1, err := json.Marshal(&v1) log.Println(string(o1), err) o2, err2 := json.

  • Go语言基础Json序列化反序列化及文件读写示例详解

    目录 概述 JSON序列化 结构体转JSON map转JSON 切片转JSON JSON反序列化 JSON转map JSON转结构体 JSON转切片 写JSON文件 map写入JSON文件 切片写入JSON文件 结构体写入JSON文件 读JSON文件 解码JSON文件为map 解码JSON文件为切片 解码JSON文件为结构体 示例 概述 JSON(JavaScript Object Notation,JavaScript对象表示法)是一种轻量级的.键值对的数据交换格式.结构由大括号'{}',中括

  • Go 结构体序列化的实现

    目录 更改JSON对象中的键 在JSON对象中隐藏结构体字段 附加内容 结构体标签string指令 本文,我们将回到之前写的showMovieHandler方法,并更新它以返回一个JSON响应,表示系统中的单个电影信息.类似于: {     "id": 123,     "title": "Casablanca",      "runtime": 102,      "genres": [        

  • C++中结构体和Json字符串互转的问题详解

    大家有没有在项目中遇到过,将一些预定义的本地结构体转换为Json字符串后,发送到网络中的情形.那我猜想下大家常规的做法:写一个函数,传入结构体的指针,然后在函数中对结构体的每一个成员根据其类型,使用Json类库的赋值方法,直接或间接创建Json子对象,组成一个内存树状结构,最后调用Json类库的方法生成字符串.这样的做法似乎比较完美,工作完成得很好,确实也挑不出什么毛病来,让我们先看看在golang中是怎么做的: type Person struct { Name string Age int

  • Golang中结构体映射mapstructure库深入详解

    目录 mapstructure库 字段标签 内嵌结构 未映射字段 Metadata 弱类型输入 逆向转换 解码器 示例 在数据传递时,需要先编解码:常用的方式是JSON编解码(参见<golang之JSON处理>).但有时却需要读取部分字段后,才能知道具体类型,此时就可借助mapstructure库了. mapstructure库 mapstructure可方便地实现map[string]interface{}与struct间的转换:使用前,需要先导入库: go get github.com/m

  • Go Struct结构体的具体实现

    目录 什么是结构体 1. 基本实例化(方法1) 2. new实例化(方法2) 3. 键值对初始化(方法3 结构体能够使用指针就使用指针) 结构体方法和接收者 encoding-json包 1. struct与json 2. struct tag 什么是结构体 Go语言中没有“类”的概念,也不支持“类”的继承等面向对象的概念.(继承,多态,封装) Go语言中通过结构体的内嵌再配合接口比面向对象具有更高的扩展性和灵活性. 1. 基本实例化(方法1) 只有当结构体实例化时,才会真正地分配内存(其实也就

  • Go语言里的结构体文法实例分析

    本文实例讲述了Go语言里的结构体文法.分享给大家供大家参考.具体分析如下: 结构体文法表示通过结构体字段的值作为列表来新分配一个结构体. 使用 Name: 语法可以仅列出部分字段.(字段名的顺序无关.) 特殊的前缀 & 构造了指向结构体文法的指针. 复制代码 代码如下: package main import "fmt" type Vertex struct {     X, Y int } var (     p = Vertex{1, 2}  // has type Ver

  • 浅谈Go语言中的结构体struct & 接口Interface & 反射

    结构体struct struct 用来自定义复杂数据结构,可以包含多个字段(属性),可以嵌套: go中的struct类型理解为类,可以定义方法,和函数定义有些许区别: struct类型是值类型. struct定义 type User struct { Name string Age int32 mess string } var user User var user1 *User = &User{} var user2 *User = new(User) struct使用 下面示例中user1和

  • Go语言指针访问结构体的方法

    本文实例讲述了Go语言指针访问结构体的方法.分享给大家供大家参考.具体分析如下: Go有指针,但是没有指针运算. 结构体字段可以通过结构体指针来访问.通过指针间接的访问是透明的. 复制代码 代码如下: package main import "fmt" type Vertex struct {     X int     Y int } func main() {     p := Vertex{1, 2}     q := &p     q.X = 1e9     fmt.P

  • 浅析C++中结构体的定义、初始化和引用

    定义:结构体(struct)是由一系列具有相同类型或不同类型的数据构成的数据集合,也叫结构. 声明一个结构体类型的形式是: 复制代码 代码如下: struct Student{      //声明一个结构体类型Student  int num;         //声明一个整形变量num  char name[20];   //声明一个字符型数组name  char sex;        //声明一个字符型变量sex  int age;         //声明一个整形变量age  float

  • Objective-C中常用的结构体NSRange,NSPoint,NSSize(CGSize),NSRect实例分析

    本文以实例详细描述了Objective-C中常用的结构体NSRange,NSPoint,NSSize(CGSize),NSRect的定义及用法,具体如下所示: 1.NSRange: NSRange的原型为 typedef struct _NSRange { NSUInteger location; NSUInteger length; } NSRange; NSMakeRange的函数: NS_INLINEz是内联函数 typedef NSRange *NSRangePointer; NS_IN

  • Swift 3.0基础学习之类与结构体

    前言 和其他语言不同的是,Swift不需要为自定义的类和结构体创建接口和实现文件.只需要创建单一文件用来创建类和结构体,其他的外部接口的代码系统会自动生成.下面这篇文章主要介绍了关于Swift 3.0类与结构体的内容,感兴趣的朋友一起来看看吧. 类和结构体区别 Swift的类和结构体具有以下相同的特点: 可以定义属性来保存值 可以定义方法来提供功能 可以定义下标来使用他们的值 可以定义初始化器来配置他们的初始化状态 可以在默认的实现上扩展他们的功能 遵从协议来提供标准的功能 类具有结构体没有的额

随机推荐