struct成员变量标签(Tag)说明
要比较详细的了解这个,要先了解一下golang的基础,在golang中,命名都是推荐都是用驼峰方式,并且在首字母大小写有特殊的语法含义:包外无法引用。但是由经常需要和其它的系统进行数据交互,例如转成json格式,存储到mongodb啊等等。这个时候如果用属性名来作为键值可能不一定会符合项目要求。
所以呢就多了反引号的内容,在golang中叫标签(Tag),在转换成其它数据格式的时候,会使用其中特定的字段作为键值。例如上例在转成json格式:
u := &User{UserId: 1, UserName: "tony"}
j, _ := json.Marshal(u)
fmt.Println(string(j))
// 输出内容:{"user_id":1,"user_name":"tony"}
如果在属性中不增加标签说明,则输出:
{"UserId":1,"UserName":"tony"}
可以看到直接用struct的属性名做键值。
其中还有一个bson的声明,这个是用在将数据存储到mongodb使用的。
其实反引号就是字符串。
Go语言中的字符串字面量使用 双引号 或 反引号 来创建 :
- 双引号用来创建 可解析的字符串字面量 (支持转义,但不能用来引用多行);
- 反引号用来创建 原生的字符串字面量 ,这些字符串可能由多行组成(不支持任何转义序列),原生的字符串字面量多用于书写多行消息、HTML以及正则表达式。
fmt.Println(`raw string`)
struct成员变量标签(Tag)获取
那么当我们需要自己封装一些操作,需要用到Tag中的内容时,咋样去获取呢?这边可以使用反射包(reflect)中的方法来获取:
t := reflect.TypeOf(u)
field := t.Elem().Field(0)
fmt.Println(field.Tag.Get("json"))
fmt.Println(field.Tag.Get("bson"))
完整代码如下: package main import ( "encoding/json" "fmt" "reflect" ) func main() { type User struct { UserId int `json:"user_id" bson:"user_id"` UserName string `json:"user_name" bson:"user_name"` } // 输出json格式 u := &User{UserId: 1, UserName: "tony"} j, _ := json.Marshal(u) fmt.Println(string(j)) // 输出内容:{"user_id":1,"user_name":"tony"} // 获取tag中的内容 t := reflect.TypeOf(u) field := t.Elem().Field(0) fmt.Println(field.Tag.Get("json")) // 输出:user_id fmt.Println(field.Tag.Get("bson")) // 输出:user_id }
json可以加上omitempty
tag中如果带有"omitempty"选项,那么如果该字段值为空,就不会输出到JSON串中
UserName string `json:"user_name,omitempty" bson:"user_name" `
u = &User{UserId: 1}
没加omit:empty {"user_id":1,"user_name":""}
加了后
{"user_id":1}
如何定义获取 Tag ?
Tag 由反引号包含,由一对或几对的键值对组成,通过空格来分割键值。格式如下
`key01:"value01" key02:"value02" key03:"value03"`
定义完后,如何从结构体中,取出 Tag 呢?
答案就是,我们上一节学过的 "反射"。
获取 Tag 可以分为三个步骤:
获取字段 field
获取标签 tag
获取键值对 key:value
type Account struct { // Id的值会进行二次JSON编码 Id int64 `json:"id"` Account string `json:"account,omitempty" orm:"size(48)"` // 账号 PassWord string `json:"-" orm:"size(32)"` // 密码 //设置字段的长度 Phone string `json:"phone" orm:"size(16)"` // 手机号 // 如果 LastLoginTime为空,则不输出到JSON串中 LastLoginTime int64 `json:"lastlogintime,omitempty"` // 设置一对一关系同时含有json输出格式 Score *Score `json:"score,omitempty" orm:"rel(one)"` // 账号财富 // 设置一对多的反向关系 PaymentLog []*PaymentLog `json:"-" orm:"reverse(many)"` // 设置一对一反向关系(可选) User *Userinfos `orm:"reverse(one)"` //设置多对多关系 Tags []*Tag `orm:"rel(m2m)"` //设置反向多对多关系 Posts []*Post `orm:"reverse(many)"` }
字段的tag是"-",那么这个字段不会输出到JSON
tag中如果带有"omitempty"选项,那么如果该字段值为空,就不会输出到JSON串中
如果字段类型是bool, string, int, int64等,而tag中带有",string"选项,那么这个字段在输出到JSON的时候会把该字段对应的值转换成JSON字符串