结构体标签

标签定义

Tag是结构体在编译阶段关联到成员的元信息字符串,在运行的时候通过反射的机制读取出来。

结构体标签由一个或多个键值对组成。键与值使用冒号分隔,值用双引号括起来。键值对之间使用一个空格分隔,具体的格式如下:

 1`key1:"value1" key2:"value2" key3:"value3"...`  

key会指定反射的解析方式,如下: json(JSON标签) orm(Beego标签)、gorm(GORM标签)、bson(MongoDB标签)、form(表单标签)、binding(表单验证标签)

标签选项

 1type Student struct {
 2    ID   int     `json:"-"`            
 3    Name string  `json:name,omitempy`  
 4    Age  int     `json:age,string`     
 5}

注:encoding/json官方文档

json标签

JSON说明

JSON数组可以用于编码Go语言的数组slice;由于JSON对象是一个字符串到值的映射,写成一系列的name:value对形式,因此JSON的对象类型可以用于编码Go语言的map结构体

将Go语言中结构体slice转为JSON的过程叫编组(marshaling),编组通过json.Marshal函数完成。在编码时,默认使用Go语言结构体的成员名字作为JSON的对象(通过reflect反射技术)。只有导出的结构体成员才会被编码。

如果在结构体slice编码成JSON的时候使用自定义的成员名,可以使用结构体成员Tag来实现。

JSON标签

 1type User struct {
 2    ID   int `json:"id"`  
 3    Name string           
 4    age  int              
 5}

json为键名的标签对应的值用于控制encoding/json包的编码和解码的行为,并且encoding/...下面其它的包也遵循这个约定。

标签选项使用说明
-字段不进行序列化 例:json:"-"
omitempy类型零值或空值,序列化时忽略该字段 例:json:",omitempy" 字段名省略的话用结构体字段名
type重新指定字段类型 例:json:"age,string"

gorm标签

模型定义

模型是标准的 struct,由基本数据类型以及实现了 Scanner 和 Valuer 接口的自定义类型及其指针或别名组成。

GORM 定义一个 gorm.Model 结构体,如下所示:

 1type Model struct {
 2  ID        uint           `gorm:"primaryKey"`
 3  CreatedAt time.Time
 4  UpdatedAt time.Time
 5  DeletedAt gorm.DeletedAt `gorm:"index"`
 6}

gorm为键名的标签遵循GORM的解析规则,GORM支持如下tag,tag名大小写不敏感,建议使用camelCase风格,多个标签定义用分号(;)分隔

[知识点] Gorm建表时 AUTO_INCREMENT 不生效的问题

 1
 2Id  uint64 `gorm:"column:id;primaryKey;type:bigint(20);autoIncrement;comment:'主键'"`
 3
 4Id  uint64 `gorm:"column:id;type:bigint(20);autoIncrement;comment:'主键'"`
 5
 6Id  uint64 `gorm:"column:id;autoIncrement;comment:'主键'"` 
标签选项使用说明
column指定 db 列名
type列数据类型,推荐使用兼容性好的通用类型,例如:所有数据库都支持 bool、int、uint、float、string、time、bytes 并且可以和其他标签一起使用,例如:not nullsizeautoIncrement… 像 varbinary(8) 这样指定数据库数据类型也是支持的。在使用指定数据库数据类型时,它需要是完整的数据库数据类型,如:MEDIUMINT UNSIGNED not NULL AUTO_INCREMENT
size指定列大小,例如:size:256
primaryKey指定列为主键
unique指定列为唯一
default指定列的默认值,字符串默认值用单引号,例:gorm:"default:'cn'"
precision指定列的精度
scale指定列大小
not null指定列为 NOT NULL
autoIncrement指定列为自动增长,不可与primaryKeytype同时使用否则不生效,看上面知识点部分
autoIncrementIncrement自动步长,控制连续记录之间的间隔
embedded嵌套字段
embeddedPrefix嵌入字段的列名前缀
autoCreateTime创建时追踪当前时间,对于 int 字段,它会追踪秒级时间戳,您可以使用 nano/milli 来追踪纳秒、毫秒时间戳,例如:autoCreateTime:nano
autoUpdateTime创建/更新时追踪当前时间,对于 int 字段,它会追踪秒级时间戳,您可以使用 nano/milli 来追踪纳秒、毫秒时间戳,例如:autoUpdateTime:milli
index根据参数创建索引,多个字段使用相同的名称则创建复合索引,查看 索引 获取详情
uniqueIndex与 index 相同,但创建的是唯一索引
check创建检查约束,例如 check:age > 13,查看 约束 获取详情
<-设置字段写入的权限, <-:create 只创建、<-:update 只更新、<-:false 无写入权限、<- 创建和更新权限
->设置字段读的权限,->:false 无读权限
-忽略该字段,- 无读写权限
comment迁移时为字段添加注释

示例:

 1
 2type Content struct {
 3    Model
 4    NewsId   uint64  `gorm:"column:news_id"`
 5    Content  string  `gorm:"column:content"`
 6}

知识点 自定义唯一索引

 1
 2
 3Email string `gorm:"column:email;type:varchar(50);uniqueIndex:uidx_email"` 
 4Email string `gorm:"column:email;type:varchar(50);index:idx_email"`        
 5
 6
 7Email string `gorm:"column:email;type:varchar(50);unique"`
 8
 9Email string `gorm:"column:email;type:varchar(50)" sql:"unique_index:uidx_email"`

知识点 自动更新时间

GORM约定使用CreatedAtUpdatedAt追踪创建/更新时间。如果定义了这种字段,且默认值为零值,GORM在创建、更新时会自动填充当前时间。要使用不同名称的字段,您可以配置autoCreateTimeautoUpdateTime标签,如果想要保存 UNIX(毫/纳)秒时间戳,而不是 time,只需简单地将 time.Time 修改为 int 即可,毫/纳秒参数可以看上面表格示例。

 1
 2type User struct { 
 3    
 4    Updated   int64 `gorm:"autoUpdateTime:nano"` 
 5    
 6    Updated   int64 `gorm:"autoUpdateTime:milli"` 
 7    
 8    Created   int64 `gorm:"autoCreateTime"` 
 9    
10    CreatedAt time.Time 
11    
12    UpdatedAt int      
13}

注:GORM模型 官方文档

关联标签

GORM的关联类型有多重类型:belongs tohas onehas manymany to many具体结构体定义可参考问文档,关联模式使用的标签选项如下所示:

标签选项使用说明
foreignKey指定当前模型的列作为连接表的外键 例:gorm:"foreignKey:FieldId" 其中FieldID是外键字段名
references指定引用表的列名,其将被映射为连接表外键
polymorphic指定多态类型,比如模型名
polymorphicValue指定多态值、默认表名
many2many指定连接表表名
joinForeignKey指定连接表的外键列名,其将被映射到当前表
joinReferences指定连接表的外键列名,其将被映射到引用表
constraint关系约束,例如:OnUpdateOnDelete

示例:

 1
 2type News struct {
 3    Model
 4    Title   string   `gorm:"column:title;type:string;not null,default:''"`
 5    Content Content  `gorm:"foreignKey:NewsId" json:"content"` 
 6}

注:GORM关联模型 官方文档

form标签

Gin中提供了模型绑定,将表单数据和模型进行绑定,方便参数校验和使用。

模型绑定

 1
 2type LoginForm struct {
 3    Email     string    `form:"emial"`    
 4    Password  string    `form:"password"`
 5}
 6
 7type Email struct {
 8    Email       string
 9    Password    string
10}

通过 form:“email” 对表单email数据进行绑定。然后通过Bind()、ShouldBind()等方法获取参数值。

 1func EmailLogin (c *gin.Context) {
 2    var email LoginForm
 3    if err := c.ShouldBind(&email); err != nil {
 4        ...
 5    }
 6    
 7    args := Email {
 8        Email:     email.Email,
 9        Password:  email.Password,
10    }
11    
12    ...
13}

binding标签

Gin对于数据的校验使用的是 validator.v10 ,该包提供多种数据校验方法,通过binding:""标签来进行数据校验。

我们对上面的表单模型添加数据校验标签如下:

 1type LoginForm struct {
 2    Email     string    `form:"emial" binding:"email"`    
 3    Password  string    `form:"password" binging:"required,min=6,max=10"`
 4}

特殊符号说明:

  • 逗号(,):分隔多个标签选项,逗号之间不能有空格,否则panic;
  • 横线(-):跳过该字段不做校验;
  • 竖线(|):使用多个选项,满足其中一个即可。

binding标签选项:

必需校验

标签选项使用说明示例
required表示该字段值必输设置,且不能为默认值binding:required
omitempty如果字段未设置,则忽略它binding:reqomitemptyuired

范围校验

范围验证: 切片、数组和map、字符串,验证其长度;数值,验证大小范围

标签选项使用说明示例
len参数值等于给定值binding:"len=8"等于8
ne不等于binding:"ne=8"不等于8
max最大值,小于等于参数值binding:"max=8"小于等于8
min最小值,大于等于参数值binding:"min=8"大于等于8
lte参数值小于等于给定值binding:"lte=8"小于等于8
gte参数值大于等于给定值binding:"gte=8"大于等于8
lt参数值小于给定值binding:"lt=8"小于8
gt参数值大于给定值binding:"gt=8"大于8
oneof参数值只能是枚举值中的一个,值必须是数值或字符串,以空格分隔,如果字符串中有空格,将字符串用单引号包围binding:"oneof=red green"

示例:

 1type User struct {
 2    Name string `form:"name" binding:"required,min=1,max=10"`
 3    Age  unit8  `form:"age" binding:"lte=150,gte=0"`
 4    sex  string `form:"sex" binding:"oneof=male female"`
 5}

注:文档地址

字符串校验

标签选项使用说明示例
contains参数值包含设置子串binding:"contains=tom"是否包含tom字符串
excludes参数值不包含设置子串binding:"excludes=tom"是否不包含tom字符串
startswith字符串前缀binding:"startswith=tom"是否以tom开头
endswith字符串前缀binding:"endswith=tom"是否以tom结尾

示例:

 1type User struct {
 2    Name string `form:"name" binding:"required,contains=ac,endswith=ck"`
 3}

注:文档地址

字段校验

标签选项使用说明
eqcsfield跨不同结构体字段相等,比如struct1 field1 是否等于struct2 field2
necsfield跨不同结构体字段不相等
eqfield同一结构体字段相等验证,例如:输入两次密码
nefield同一结构体字段不相等验证
gtefield大于等于同一结构体字段
ltefield小于等于同一结构体字段

示例:

 1
 2type Struct1 struct { 
 3    Field1 string `validate:eqcsfield=Struct2.Field2` 
 4    Struct2 struct { 
 5        Field2 string 
 6    } 
 7}
 8
 9type Email struct { 
10    Email  string `validate:"lte=4"` 
11    Pwd    string `validate:"min=10"` 
12    Pwd2   string `validate:"eqfield=Pwd"`
13}
14
15type User struct { 
16    Name     string `validate:"lte=4"` 
17    Age      int `validate:"min=20"` 
18    Password string `validate:"min=10,nefield=Name"`
19}

其他校验

标签选项使用说明示例
ip合法IP地址校验binding:"ip"
email合法邮箱校验binding:"email"
url合法的URLbinding:"url"
uri合法的URIbinding:"uri"
uuiduuid验证binding:"uuid"
datetime合法时间格式值校验binding:"datetime=2006-01-02"
jsonJSON数据验证validate:"json"
numeric数值验证 正则:^[-+]?[0-9]+(?:\\.[0-9]+)?$validate:"numeric"
number整数验证 正则:^[0-9]+$validate:"number"
alpha字母字符串验证 正则:^[a-zA-Z]+$validate:"alpha"
alphanum字母数字字符串验证 正则:^[a-zA-Z0-9]+$validate:"alphanum"
asciiAscii 字符验证validate:"ascii"

示例:

 1type User struct { 
 2    Name     string  `validate:"required,min=1,max=10"` 
 3    Email    int     `validate:"required,email"`
 4    birthday string  `validate:"datetime=2006-01-02"`
 5    Pwd      string  `validate:"required,alphanum"`
 6    Score    srring  `validate:"numeric"`
 7}

注:文档地址

ini标签

在使用go-ini库操作.ini配置文件的时候,如果需要将配置文件字段映射到结构体变量,如果键名与字段名不相同,那么需要在结构标签中指定对应的键名。标准库encoding/jsonencoding/xml解析时可以将键名app_name对应到字段名AppName,而go-ini库不可以,所以需要在结构体标签指定对应键名。

 1
 2app_name  = awesome web
 3log_level = DEBUG
 1
 2type Config struct {
 3  AppName   string `ini:"app_name"`  
 4  LogLevel  string `ini:"log_level"`
 5}

参考文章

个人笔记记录 2021 ~ 2025