為什么 Go 語言 struct 要使用 Tags
在 Go 語言中,struct 是一種常見的數(shù)據(jù)類型,它可以用來表示復(fù)雜的數(shù)據(jù)結(jié)構(gòu)。在 struct 中,我們可以定義多個字段,每個字段可以有不同的類型和名稱。
除了這些基本信息之外,Go 還提供了 struct tags,它可以用來指定 struct 中每個字段的元信息。
在本文中,我們將探討為什么 Go 語言中需要使用 struct tags,以及 struct tags 的使用場景和優(yōu)勢。
struct tags 的使用
struct tags 使用還是很廣泛的,特別是在 json 序列化,或者是數(shù)據(jù)庫 ORM 映射方面。
在定義上,它以 key:value 的形式出現(xiàn),跟在 struct 字段后面,除此之外,還有以下幾點(diǎn)需要注意:
使用反引號
在聲明 struct tag 時,使用反引號 ` 包圍 tag 的值,可以防止轉(zhuǎn)義字符的影響,使 tag 更容易讀取和理解。例如:
避免使用空格
在 struct tag 中,應(yīng)該避免使用空格,特別是在 tag 名稱和 tag 值之間。使用空格可能會導(dǎo)致編碼或解碼錯誤,并使代碼更難以維護(hù)。例如:
避免重復(fù)
在 struct 中,應(yīng)該避免重復(fù)使用同一個 tag 名稱。如果重復(fù)使用同一個 tag 名稱,編譯器可能會無法識別 tag,從而導(dǎo)致編碼或解碼錯誤。例如:
使用標(biāo)準(zhǔn)化的 tag 名稱
為了使 struct tag 更加標(biāo)準(zhǔn)化和易于維護(hù),應(yīng)該使用一些標(biāo)準(zhǔn)化的 tag 名稱。
例如,對于序列化和反序列化,可以使用 json、xml、yaml? 等;對于數(shù)據(jù)庫操作,可以使用 db。
其中,Password? 字段后面的 - 表示忽略該字段,也就是說該字段不會被序列化或反序列化。
多個 tag 值
如果一個字段需要指定多個 tag 值,可以使用 , 將多個 tag 值分隔開。例如:
其中 omitempty 表示如果該字段值為空,則不序列化該字段。
struct tags 的原理
Go 的反射庫提供了一些方法,可以讓我們在程序運(yùn)行時獲取和解析結(jié)構(gòu)體標(biāo)簽。
介紹這些方法之前,先來看看 reflect.StructField ,它是描述結(jié)構(gòu)體字段的數(shù)據(jù)類型。定義如下:
結(jié)構(gòu)體中還有一些其他字段,被我省略了,只保留了和本文相關(guān)的。
在結(jié)構(gòu)體的反射中,我們經(jīng)常使用 reflect.TypeOf? 獲取類型信息,然后使用 Type.Field? 或 Type.FieldByName()? 獲取結(jié)構(gòu)體字段的 reflect.StructField?,然后根據(jù) StructField 中的信息做進(jìn)一步處理。
例如,可以通過 StructField.Tag.Get 方法獲取結(jié)構(gòu)體字段的標(biāo)簽值。
下面看一段代碼:
運(yùn)行以上代碼,輸出結(jié)果如下:
struct tags 的優(yōu)勢
使用 struct tag 的主要優(yōu)勢之一是可以在運(yùn)行時通過反射來訪問和操作 struct 中的字段。
比如在 Go Web 開發(fā)中,常常需要將 HTTP 請求中的參數(shù)綁定到一個 struct 中。這時,我們可以使用 struct tag 指定每個字段對應(yīng)的參數(shù)名稱、驗(yàn)證規(guī)則等信息。在接收到 HTTP 請求時,就可以使用反射機(jī)制讀取這些信息,并根據(jù)信息來驗(yàn)證參數(shù)是否合法。
另外,在將 struct 序列化為 JSON 或者其他格式時,我們也可以使用 struct tag 來指定每個字段在序列化時的名稱和規(guī)則。
此外,使用 struct tag 還可以提高代碼的可讀性和可維護(hù)性。在一個大型的項(xiàng)目中,struct 中的字段通常會包含很多不同的元信息,比如數(shù)據(jù)庫中的表名、字段名、索引、驗(yàn)證規(guī)則等等。
如果沒有 struct tag,我們可能需要將這些元信息放在注釋中或者在代碼中進(jìn)行硬編碼。這樣會讓代碼變得難以維護(hù)和修改。而使用 struct tag 可以將這些元信息與 struct 字段緊密關(guān)聯(lián)起來,使代碼更加清晰和易于維護(hù)。
常用的 struct tags
在 Go 的官方 wiki 中,有一個常用的 struct tags 的庫的列表,我復(fù)制在下面了,感興趣的同學(xué)可以看看源碼,再繼續(xù)深入學(xué)習(xí)。
Tag | Documentation |
xml | |
json | |
asn1 | |
reform | |
dynamodb | ??https://docs.aws.amazon.com/sdk-for-go/api/service/dynamodb/dynamodbattribute/#Marshal?? |
bigquery | |
datastore | |
spanner | |
bson | https://pkg.go.dev/labix.org/v2/mgo/bson, https://pkg.go.dev/go.mongodb.org/mongo-driver/bson/bsoncodec |
gorm | |
yaml | |
toml | |
validate | |
mapstructure | |
parser | |
protobuf | |
db | |
url | |
feature |
以上就是本文的全部內(nèi)容,如果覺得還不錯的話歡迎點(diǎn)贊,轉(zhuǎn)發(fā)和關(guān)注,感謝支持。
參考文章:
- https://github.com/golang/go/wiki/Well-known-struct-tags