PHP转Go之玩转Golang结构体的tag

结构体的tag是Golang区别于PHP独有的概念,本文会详细介绍Go中tag的定义、用法及常用场景

本文概要

  • 结构体 tag 的定义
  • 结构体 tag 的读取
  • 结构体 tag 的常用场景

结构体 tag 的定义

在Go语言中,结构体标签是通过在结构体定义的字段后面添加一个字符串来定义的。这个字符串被称为标签(tag),其中包含了一些键-值对,以键:值的形式表示。例如:

Golang 复制代码
type Person struct { 
    Name string `json:"name" xml:"name" csv:"name"` 
    Age int `json:"age" xml:"age" csv:"age"` 
    Address string `json:"address" xml:"address" csv:"address"` 
}

如何读取结构体 tag

tag的读取利用的是golagn的反射机制

Golang 复制代码
package main

import (
	"fmt"
	"reflect"
)

type Person struct {
	Name string `json:"name" xml:"person_name"`
	Age  int    `json:"age" xml:"person_age"`
}

func main() {
	p := Person{Name: "John Doe", Age: 30}

	// 获取结构体的反射类型
	t := reflect.TypeOf(p)

	// 遍历结构体的字段
	for i := 0; i < t.NumField(); i++ {
		field := t.Field(i)
		tag := field.Tag.Get("json") // 获取json标签
		fmt.Printf("Field: %s, Tag: %s\n", field.Name, tag)
	}
}
// Output:
Field: Name, Tag: name
Field: Age, Tag: age

结构体 tag 的常用场景

  1. 序列化和反序列化

比如上面的tag中包含的json tag,当使用json.Marshal时,会读取对应字段定义的json tag,然后将json对象中的key赋值为该值。

需要特别注意,对于以小写开头的字段,序列化及反序列化时是忽略的;另外对于即使没定义json tag的字段序列化时通常也会默认以字段名赋值到json的键名上

  1. 验证和过滤

这个是非常常见的需求,比如前端传过来的参数都需要校验下;这个验证比较常用的库就是 go-playground/validator,只需要在结构体tag上完成验证规则标记,即可快速书进行验证

Golang 复制代码
type User struct {  
    Name     string `validate:"required"`  
    Age      int    `validate:"min=1,max=120"`  
    Email    string `validate:"required,email"`  
    Password string `validate:"required,min=6"`  
} 
  1. 数据库列映射

常用的数据库,比如gorm or xorm均是利用tag完成数据字段的对应

Golang 复制代码
// Gorm
type User struct {  
    Name  string `gorm:"column:username"`  
    Age   int    `gorm:"column:age"`  
    Email string `gorm:"column:email"`  
} 
// Xorm
type User struct {  
    Name     string `xorm:"'username'"`  
    Age      int    `xorm:"'age'"`  
    Email    string `xorm:"'email'"`  
}  

tag 的用处自然不止这么点,利用tag可以完成更多的程序的自动化,比如标记某个字段是否记录日志等等,tag也可以完全自定义用途

相关推荐
喜欢敲代码的程序员26 分钟前
Spring Boot中请求参数读取方式
java·spring boot·后端·spring
中烟创新1 小时前
DeepSeek部署实战:模型对比、部署优化与应用场景解析
前端·后端
亚洲第一中锋_哈达迪2 小时前
详解缓存淘汰策略:LFU
后端·缓存·golang
灰小猿2 小时前
多级@JsonTypeInfo和@JsonSubTypes注解使用详解及场景分析
java·后端·mysql·spring·spring cloud
子林super2 小时前
PostgreSQL主从切换后时间线修复操作手册
后端
程序员鱼皮2 小时前
Stack Overflow,彻底凉了!
前端·后端·计算机·程序员·互联网
neoooo2 小时前
Spring Boot 中的 synchronized(this):到底锁住了谁?
java·spring boot·后端
京东云开发者2 小时前
基于业务知识和代码库增强的大模型生成代码实践
后端
贵州数擎科技有限公司2 小时前
LangChain 快速构建你的第一个 LLM 应用
前端·后端