詳解Go語言中Validator庫的使用方法和用途
1. 引言
github.com/go-playground/validator
是一個 Go 語言的庫,用于對結(jié)構(gòu)體字段進(jìn)行驗證。它提供了一種簡單而靈活的方式來定義驗證規(guī)則,并在驗證過程中檢查結(jié)構(gòu)體字段是否滿足這些規(guī)則。這個庫可以用于驗證各種數(shù)據(jù),包括從用戶輸入到 API 請求中的數(shù)據(jù),以確保數(shù)據(jù)的完整性和有效性。
在這篇文章中,我們將從一個簡單的問題出發(fā),帶你了解 Validator
庫的用途,也會介紹Validator
的基本使用,同時也會介紹Validator
能夠給我們帶來的優(yōu)點。
2. 問題引入
在平常開發(fā)過程中,不管是Web應(yīng)用程序來接收頁面請求,還是創(chuàng)建一個服務(wù)來接收其他服務(wù)的請求,不可避免的,我們都需要檢查請求參數(shù)是否合法,是否有效。
假設(shè)我們開發(fā)一個用戶注冊功能的 Web 應(yīng)用程序。用戶在注冊頁面上提供了以下信息:用戶名、電子郵件地址、密碼和確認(rèn)密碼。那么我們必須編寫下述代碼,保證用戶輸入信息合法性,如下:
type User struct { Username string Email string } func (u *User) checkUserIsInvalid() error { // 檢查用戶名長度是否合法 if len(user.Username) < 3 || len(user.Username) > 20 { return errors.New("Invalid username length") } // 檢查電子郵件地址是否合法 if !isValidEmail(user.Email) { return errors.New("Invalid email address") } return nil } func registerUser(user User) error { // 檢查輸入是否合法 err := user.checkUserIsInvalid() if err != nil { return errors.New("Invalid Input") } // 用戶注冊邏輯... return nil }
這里的實現(xiàn)并沒有太大的問題。但是如果程序中有20個地方,都檢查了用戶名長度是否合法,如果這個驗證邏輯更復(fù)雜一點,那就不太合理了,這里的一個做法是將驗證邏輯抽取為一個函數(shù),示例如下:
func checkUserNameIsValid(username string) bool{ if len(username) < 3 || len(username) > 20 { return false } return true }
然后用到這段邏輯的,直接調(diào)用該函數(shù)即可,不需要再重復(fù)實現(xiàn),這個也能夠解決一部分場景的問題。但是假想一下,如果我們的驗證邏輯不像上面那么簡單,而是涉及到多個字段的組合驗證,類型轉(zhuǎn)換,嵌套結(jié)構(gòu)體的場景,這個時候我們的驗證邏輯會非常復(fù)雜。
比如我們需要實現(xiàn)一個嵌套結(jié)構(gòu)體的校驗邏輯,此時我們需要遍歷每一個字段,可能會有非常深的if...else代碼,亦或者比較深層次的函數(shù)調(diào)用,這個復(fù)雜邏輯不管是實現(xiàn)還是后續(xù)的閱讀,都會花費我們大量的精力。
回歸到我們的訴求,其實我們并不是很關(guān)心嵌套了多少層結(jié)構(gòu)體,我們更關(guān)注的是針對某一個 字段/值,其值是否滿足我們的預(yù)期。那有沒有辦法,做到我們實現(xiàn)一個驗證邏輯,通過某種手段作用到目標(biāo)字段,而不需要去關(guān)注具體的數(shù)據(jù)結(jié)構(gòu),這樣子既能做到驗證邏輯的復(fù)用,同時也避免了對復(fù)雜數(shù)據(jù)結(jié)構(gòu)的解析,從而簡化我們的驗證邏輯。
其實還真有,當(dāng)前存在大量的驗證庫,能夠幫助我們實現(xiàn)數(shù)據(jù)驗證。接下來我們就來了解下Go語言中的Validator
庫,其能夠讓我們專注于驗證邏輯的編寫,而不需要考慮邏輯的復(fù)用以及復(fù)雜數(shù)據(jù)結(jié)構(gòu)的處理等許多問題,同時在某種程度上也提高了代碼的可讀性。
3. Validator 的基本使用
Validator
是基于標(biāo)簽來實現(xiàn)的,我們只需要在結(jié)構(gòu)體的字段上使用 validate
標(biāo)簽,然后設(shè)置標(biāo)簽值,每一個標(biāo)簽值代表一個驗證規(guī)則。這些標(biāo)簽值將告訴 validator
結(jié)構(gòu)體的字段應(yīng)該滿足哪些條件,然后通過調(diào)用Validator
提供的 API
,便能夠?qū)崿F(xiàn)數(shù)據(jù)的校驗。
下面我們通過一個簡單的例子來進(jìn)行說明,幫助我們快速入門Validator
庫的使用:
type User struct { FirstName string `validate:"required"` LastName string `validate:"required"` Age uint8 `validate:"gte=20,lte=60"` Email string `validate:"required,email"` } func main() { validate = validator.New() user := &User{ FirstName: "Badger", LastName: "Smith", Age: 18, Email: "Badger.Smith@gmail.com", } // returns nil or ValidationErrors ( []FieldError ) err := validate.Struct(user) if err != nil { fmt.Println(err) } }
上面例子中,我們定義了一個 User
結(jié)構(gòu)體,包含了不同類型的字段,每個字段都通過validate
標(biāo)簽定義一些驗證規(guī)則。
其中FirstName
和 LastName
都設(shè)置了 required
規(guī)則,Age
設(shè)置了gte=0
和 lte=130
規(guī)則,Email
則設(shè)置了required
和 email
兩個規(guī)則。其中required
,gte
,lte
和 email
規(guī)則是 Validator
庫自帶的校驗規(guī)則,可以直接設(shè)置。
在結(jié)構(gòu)體設(shè)置好驗證規(guī)則后,在main
函數(shù)中通過New
方法創(chuàng)建一個 Validate
實例,然后通過調(diào)用Struct
方法,便會自動根據(jù)結(jié)構(gòu)體標(biāo)簽設(shè)置的規(guī)則對對象的值進(jìn)行驗證。如果驗證通過,將返回nil
,否則會返回一個ValidationErrors
類型的錯誤對象,其中包含驗證失敗的詳細(xì)信息。
比如上面Age
字段不滿足條件,此時user
對象將不能通過校驗,會返回對應(yīng)的錯誤信息,如下:
Key: 'User.Age' Error:Field validation for 'Age' failed on the 'gte' tag
4. Validator優(yōu)點
如果我們使用 Validator
庫,邏輯就可以抽取出來為一個公共的驗證庫,然后每一個驗證邏輯對應(yīng)一個驗證規(guī)則名,這個Validator
庫有支持,后續(xù)會講述到。
然后在結(jié)構(gòu)體中,使用validate
標(biāo)簽指定需要的驗證規(guī)則,這樣子我們就不需要待驗證數(shù)據(jù)的數(shù)據(jù)結(jié)構(gòu),也復(fù)用了驗證規(guī)則,同時將驗證規(guī)則與字段綁定到一起,也提高了代碼的可讀性。
通過使用Validator
庫,我們能夠回歸到核心關(guān)注的內(nèi)容,驗證傳入數(shù)據(jù)的合法性, 而不是去解析數(shù)據(jù)結(jié)構(gòu),代碼復(fù)用等一系列復(fù)雜的事情,把這些復(fù)雜的事情交給 Validator
幫我們做。
5. 總結(jié)
本文介紹了 Go 語言中的 github.com/go-playground/validator
庫,該庫用于對結(jié)構(gòu)體字段進(jìn)行驗證。文章從一個簡單的問題出發(fā),引入了Validator
庫的使用。
之后介紹了 Validator
庫的基本使用,包括如何創(chuàng)建驗證實例、執(zhí)行驗證以及處理驗證錯誤。通過示例代碼,演示了如何使用標(biāo)簽來設(shè)置驗證規(guī)則,以及如何通過 Validator
庫簡化數(shù)據(jù)驗證過程,提高代碼的可讀性和可維護(hù)性。
總的來說,在比較復(fù)雜的場景,通過使用Validator
庫,我們可以專注于驗證邏輯的編寫,而不必?fù)?dān)心數(shù)據(jù)結(jié)構(gòu)的解析和重復(fù)的驗證代碼,能夠很好得提高代碼的可讀性和可維護(hù)性。
以上就是詳解Go語言中Validator庫的使用方法和用途的詳細(xì)內(nèi)容,更多關(guān)于Go Validator庫使用的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Golang 類型轉(zhuǎn)換的實現(xiàn)(斷言、強制、顯式類型)
將一個值從一種類型轉(zhuǎn)換到另一種類型,便發(fā)生了類型轉(zhuǎn)換,在go可以分為斷言、強制、顯式類型轉(zhuǎn)換,本文就詳細(xì)的介紹一下這就幾種轉(zhuǎn)換方式,具有一定的參考價值,感興趣的可以了解一下2023-09-09Go?channel實現(xiàn)批量讀取數(shù)據(jù)
Go中的?channel?其實并沒有提供批量讀取數(shù)據(jù)的方法,需要我們自己實現(xiàn)一個,使用本文就來為大家大家介紹一下如何通過Go?channel實現(xiàn)批量讀取數(shù)據(jù)吧2023-12-12Go語言日志內(nèi)聚復(fù)用及gjson踩坑記錄分享
這篇文章主要為大家介紹了Go語言日志內(nèi)聚復(fù)用及gjson踩坑記錄分享,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-06-06解決Go中攔截HTTP流數(shù)據(jù)時字段丟失的問題
在開發(fā)高并發(fā)的Web應(yīng)用時,尤其是在處理HTTP代理和流數(shù)據(jù)攔截的場景下,遇到數(shù)據(jù)丟失的問題并不罕見,最近,在一個項目中,我遇到了一個棘手的問題:在攔截并轉(zhuǎn)發(fā)HTTP流數(shù)據(jù)的過程中,某些數(shù)據(jù)字段因為處理過快而被丟失,所以本文給大家介紹如何解決這個問題2024-08-08