Go语言struct要使用 tags的原因解析
作者:yongxinz 发布时间:2023-08-31 09:25:19
在 Go 语言中,struct 是一种常见的数据类型,它可以用来表示复杂的数据结构。在 struct 中,我们可以定义多个字段,每个字段可以有不同的类型和名称。
除了这些基本信息之外,Go 还提供了 struct tags,它可以用来指定 struct 中每个字段的元信息。
在本文中,我们将探讨为什么 Go 语言中需要使用 struct tags,以及 struct tags 的使用场景和优势。
struct tags 的使用
struct tags 使用还是很广泛的,特别是在 json 序列化,或者是数据库 ORM 映射方面。
在定义上,它以 key:value
的形式出现,跟在 struct 字段后面,除此之外,还有以下几点需要注意:
使用反引号
在声明 struct tag 时,使用反引号 `
包围 tag 的值,可以防止转义字符的影响,使 tag 更容易读取和理解。例如:
type User struct {
ID int `json:"id" db:"id"`
Name string `json:"name" db:"name"`
Email string `json:"email" db:"email"`
}
避免使用空格
在 struct tag 中,应该避免使用空格,特别是在 tag 名称和 tag 值之间。使用空格可能会导致编码或解码错误,并使代码更难以维护。例如:
// 不规范的写法
type User struct {
ID int `json: "id" db: "id"`
Name string `json: "name" db: "name"`
Email string `json: "email" db: "email"`
}
// 规范的写法
type User struct {
ID int `json:"id" db:"id"`
Name string `json:"name" db:"name"`
Email string `json:"email" db:"email"`
}
避免重复
在 struct 中,应该避免重复使用同一个 tag 名称。如果重复使用同一个 tag 名称,编译器可能会无法识别 tag,从而导致编码或解码错误。例如:
// 不规范的写法
type User struct {
ID int `json:"id" db:"id"`
Name string `json:"name" db:"name"`
Email string `json:"email" db:"name"`
}
// 规范的写法
type User struct {
ID int `json:"id" db:"id"`
Name string `json:"name" db:"name"`
Email string `json:"email" db:"email"`
}
使用标准化的 tag 名称
为了使 struct tag 更加标准化和易于维护,应该使用一些标准化的 tag 名称。
例如,对于序列化和反序列化,可以使用 json
、xml
、yaml
等;对于数据库操作,可以使用 db
。
type User struct {
ID int `json:"id" db:"id"`
Name string `json:"name" db:"name"`
Password string `json:"-" db:"password"` // 忽略该字段
Email string `json:"email" db:"email"`
}
其中,Password
字段后面的 -
表示忽略该字段,也就是说该字段不会被序列化或反序列化。
多个 tag 值
如果一个字段需要指定多个 tag 值,可以使用 ,
将多个 tag 值分隔开。例如:
type User struct {
ID int `json:"id" db:"id"`
Name string `json:"name" db:"name"`
Email string `json:"email,omitempty" db:"email,omitempty"`
}
其中 omitempty
表示如果该字段值为空,则不序列化该字段。
struct tags 的原理
Go 的反射库提供了一些方法,可以让我们在程序运行时获取和解析结构体标签。
介绍这些方法之前,先来看看 reflect.StructField
,它是描述结构体字段的数据类型。定义如下:
type StructField struct {
Name string // 字段名
Type Type // 字段类型
Tag StructTag // 字段标签
}
结构体中还有一些其他字段,被我省略了,只保留了和本文相关的。
在结构体的反射中,我们经常使用 reflect.TypeOf
获取类型信息,然后使用 Type.Field
或 Type.FieldByName()
获取结构体字段的 reflect.StructField
,然后根据 StructField
中的信息做进一步处理。
例如,可以通过 StructField.Tag.Get
方法获取结构体字段的标签值。
下面看一段代码:
package main
import (
"fmt"
"reflect"
)
type User struct {
Name string `json:"name"`
Age int `json:"age"`
}
type Manager struct {
Title string `json:"title"`
User
}
func main() {
m := Manager{Title: "Manager", User: User{Name: "Alice", Age: 25}}
mt := reflect.TypeOf(m)
// 获取 User 字段的 reflect.StructField
userField, _ := mt.FieldByName("User")
fmt.Println("Field 'User' exists:", userField.Name, userField.Type)
// 获取 User.Name 字段的 reflect.StructField
nameField, _ := userField.Type.FieldByName("Name")
tag := nameField.Tag.Get("json")
fmt.Println("User.Name tag:", tag)
}
运行以上代码,输出结果如下:
Field 'User' exists: User {string int}
User.Name tag: "name"
struct tags 的优势
使用 struct tag 的主要优势之一是可以在运行时通过反射来访问和操作 struct 中的字段。
比如在 Go Web 开发中,常常需要将 HTTP 请求中的参数绑定到一个 struct 中。这时,我们可以使用 struct tag 指定每个字段对应的参数名称、验证规则等信息。在接收到 HTTP 请求时,就可以使用反射机制读取这些信息,并根据信息来验证参数是否合法。
另外,在将 struct 序列化为 JSON 或者其他格式时,我们也可以使用 struct tag 来指定每个字段在序列化时的名称和规则。
此外,使用 struct tag 还可以提高代码的可读性和可维护性。在一个大型的项目中,struct 中的字段通常会包含很多不同的元信息,比如数据库中的表名、字段名、索引、验证规则等等。
如果没有 struct tag,我们可能需要将这些元信息放在注释中或者在代码中进行硬编码。这样会让代码变得难以维护和修改。而使用 struct tag 可以将这些元信息与 struct 字段紧密关联起来,使代码更加清晰和易于维护。
常用的 struct tags
在 Go 的官方 wiki 中,有一个常用的 struct tags 的库的列表,我复制在下面了,感兴趣的同学可以看看源码,再继续深入学习。
Tag | Documentation |
---|---|
xml | https://pkg.go.dev/encoding/xml |
json | https://pkg.go.dev/encoding/json |
asn1 | https://pkg.go.dev/encoding/asn1 |
reform | https://pkg.go.dev/gopkg.in/reform.v1 |
dynamodb | https://docs.aws.amazon.com/sdk-for-go/api/service/dynamodb/dynamodbattribute/#Marshal |
bigquery | https://pkg.go.dev/cloud.google.com/go/bigquery |
datastore | https://pkg.go.dev/cloud.google.com/go/datastore |
spanner | https://pkg.go.dev/cloud.google.com/go/spanner |
bson | https://pkg.go.dev/labix.org/v2/mgo/bson, https://pkg.go.dev/go.mongodb.org/mongo-driver/bson/bsoncodec |
gorm | https://pkg.go.dev/github.com/jinzhu/gorm |
yaml | https://pkg.go.dev/gopkg.in/yaml.v2 |
toml | https://pkg.go.dev/github.com/pelletier/go-toml |
validate | https://github.com/go-playground/validator |
mapstructure | https://pkg.go.dev/github.com/mitchellh/mapstructure |
parser | https://pkg.go.dev/github.com/alecthomas/participle |
protobuf | https://github.com/golang/protobuf |
db | https://github.com/jmoiron/sqlx |
url | https://github.com/google/go-querystring |
feature | https://github.com/nikolaydubina/go-featureprocessing |
来源:https://www.cnblogs.com/alwaysbeta/p/17205950.html


猜你喜欢
- 实例如下:String.prototype.trim = function (char, type) { if (char) {
- 最近网上流行着一些采集程序,更多人拿着这些东西在网上叫卖,很多不太懂的人看着那些程序眼羡,其实如果你懂一些ASP,了解自动采集程序的原理后,
- 一、变量和表达式>>> 1 + 1 &n
- 前言我们经常会与文件和目录打交道,对于这些操作python提供了一个os模块,里面包含了很多操作文件和目录的函数。在写一些系统脚本或者自动化
- 以下是作者在学习Python中django框架时的学习笔记,并把测试的代码做了详细分析,最后还附上了学习心得,值得大家学习。URL配置(UR
- 在DreamWeaver中编写CSS,这种编写习惯本站(twocity.cn)并不提倡,不过由于"可视化"和操作简便,使
- 记得当时我们要跟网友天下合作,需要这样的一个鼠标划过放大的列表效果,就写了一个效果截图:演示代码:<!DOCTYPE html PUB
- python 二维列表转置def transpose(self, matrix): new_matrix = []
- MySQL作为数据库的一大主力军,到处存在于我们各种系统中,相信大家都不陌生!但是,你知道你能用不代表你知道细节,那我们就来盘点盘点其中一些
- 本文描述通过统计分析出医院信息系统需分区的表,对需分区的表选择分区键,即找出包括在你的分区键中的列(表的属性),对大型数据的管理比较有意义,
- 我见朋友可以把数据库的记录显示到列表框里去,挺实用,也想做一个。怎么做啊?这简单,代码和说明如下:dblist.asp<html>
- 在停止和开始进度条的同时,将进度条清空的动作也是常常需要用到的。具体用法如下:self.progressBar = QProgressBar
- 一种类似Flask开发的WebSocket-Server服务端框架,适用python3.X1、安装模块Pywsspip install py
- openpyxl是一个读写Excel文档的Python库,能够同时读取和修改Excel文档。openpyxl是一个开源项目,因此在使用之前需
- php开启openssl的方法,大多数情况下openssl是没有开启的,要想启用需要进行下简单的设置windows下开启方法:1: 首先检查
- Python Scrapy爬虫,听说妹子图挺火,我整站爬取了,上周一共搞了大概8000多张图片。和大家分享一下。核心爬虫代码# -*- co
- 回忆一下mysql 5.5 安装配置方法,整理mysql 5.5 安装配置教程笔记,分享给大家。MySQL下载地址:http://dev.m
- 12-24小时制编写一个程序,要求用户输入24小时制的时间,然后显示12小时制的时间。输入格式:输入在一行中给出带有中间的:符号(半角的冒号
- 一、获取Tensor神经网络在运算过程中实际上是以Tensor为格式进行计算的,我们只需稍稍改动一下forward函数即可从运算过程中抓到T
- 为什页面刷新会出现404因为vue项目中路由hash模式改为了history模式,由于hash模式时url带的#号后面是哈希值不会作为url