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字符串