在處理json格式字符串的時候,常常會看到聲明struct結構的時候,屬性的右側還有小米點括起來的內容。形如golang
type User struct { UserId int `json:"user_id" bson:"user_id"` UserName string `json:"user_name" bson:"user_name"` }
要比較詳細的瞭解這個,要先了解一下golang的基礎,在golang中,命名都是推薦都是用駝峯方式,而且在首字母大小寫有特殊的語法含義:包外沒法引用。可是由常常須要和其它的系統進行數據交互,例如轉成json格式,存儲到mongodb啊等等。這個時候若是用屬性名來做爲鍵值可能不必定會符合項目要求。mongodb
因此呢就多了小米點的內容,在golang中叫標籤(Tag),在轉換成其它數據格式的時候,會使用其中特定的字段做爲鍵值。例如上例在轉成json格式:json
u := &User{UserId: 1, UserName: "tony"} j, _ := json.Marshal(u) fmt.Println(string(j)) // 輸出內容:{"user_id":1,"user_name":"tony"}
若是在屬性中不增長標籤說明,則輸出:spa
{"UserId":1,"UserName":"tony"}
能夠看到直接用struct的屬性名作鍵值。code
其中還有一個bson的聲明,這個是用在將數據存儲到mongodb使用的。blog
那麼當咱們須要本身封裝一些操做,須要用到Tag中的內容時,如何去獲取呢?這邊能夠使用反射包(reflect)中的方法來獲取:字符串
t := reflect.TypeOf(u) field := t.Elem().Field(0) fmt.Println(field.Tag.Get("json")) fmt.Println(field.Tag.Get("bson"))
完整代碼以下:string
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
beego的ORM中也經過tag來定義參數的。class
package main import ( "fmt" "reflect" ) type Job struct { AlarmStatus *string `json:"alarm_status" name:"alarm_status"` CPUTopology string `json:"cpu_topology" name:"cpu_topology"` } func main(){ a := "abc" s := Job{&a,"hello"} st := reflect.TypeOf(s) field := st.Field(1) fmt.Println(field.Tag.Get("json"), field.Tag.Get("name")) }
package main import ( "fmt" "reflect" // 這裏引入reflect模塊 ) type User struct { Name string "user name" //這引號裏面的就是tag Passwd string "user passsword" } func main() { user := &User{"chronos", "pass"} s := reflect.TypeOf(user).Elem() //經過反射獲取type定義 for i := 0; i < s.NumField(); i++ { fmt.Println(s.Field(i).Tag) //將tag輸出出來 } }