Go微服務(wù)項目配置文件的定義和讀取示例詳解
前言

我們在寫應(yīng)用時,基本都會用到配置文件,從各種 shell 到 nginx 等,都有自己的配置文件。雖然這沒有太多難度,但是配置項一般相對比較繁雜,解析、校驗也會比較麻煩。本文就給大家講講我們是怎么簡化配置文件的定義和解析的。
場景
如果我們要寫一個 Restful API 的服務(wù),配置項大概有如下內(nèi)容:
- Host,偵聽的 IP,如果不填,默認用 0.0.0.0
- Port,偵聽的端口,必填,只能是數(shù)字,大于等于80,小于65535
- LogMode,日志模式,只能選 file 或者 console
- Verbose,看是否輸出詳細日志,可選,默認為 false
- MaxConns,允許的最大并發(fā)連接數(shù),默認 10000
- Timeout,超時設(shè)置,默認 3s
- CpuThreshold,設(shè)置 CPU 使用率觸發(fā)系統(tǒng)降載的閾值,默認 900,1000m 表示 100%
之前我們用 json 做配置文件,但是 json 有個問題,無法加注釋,所以我們后來切換到了 yaml 格式。
接下來讓我們看看借助 go-zero 怎么來方便的的定義和解析這樣的配置文件~
定義配置
首先,我們需要將上述配置需求定義到 Go 結(jié)構(gòu)體里,如下:
RestfulConf struct {
Host string `json:",default=0.0.0.0"`
Port int `json:",range=[80,65535)"`
LogMode string `json:",options=[file,console]"`
Verbose bool `json:",optional"`
MaxConns int `json:",default=10000"`
Timeout time.Duration `json:",default=3s"`
CpuThreshold int64 `json:",default=900,range=[0:1000]"`
}
可以看到,我們對每個配置項都有一定的定義和限制,其中一些定義如下:
- default,配置沒填的話,使用該默認值,可以看到其中的 3s 會自動解析成 time.Duration 類型
- optional,此項可以不配置,沒有的話,用類型零值
- range,限定數(shù)字類型,需要在給定的范圍內(nèi)
- options,限制配置的值只能是給出的這幾個之一
并且,一些屬性可以疊加使用,比如:
- default 和 range 一起使用,就可以既增加了范圍限制,又提供了默認值
- default 和 options 一起使用,就可以既增加了可選項限制,又提供了默認值
配置文件
因為我們在定義配置的時候,給了很多的默認值,還有使用 optional 指定為可選,所以我們的配置文件里的配置項就相對比較少了,能用默認值的就不用寫了,如下:
# 因為很多都有默認值,所以只需要寫需要指定值和沒有默認值的
Port: 8080
LogMode: console
# 可以讀取環(huán)境變量的值
MaxBytes: ${MAX_BYTES}
這里有個注意點,如果配置項的 value 全部是數(shù)字,而你定義的配置類型是 string,比如有人測試密碼經(jīng)常用 123456,但是密碼一般會定義為 string,配置就要寫成如下(只是舉個例子哈,密碼一般不建議裸寫到配置文件里):
Password: "123456"
這里的雙引號不能少,少了會報 type mismatch 之類的錯誤,因為yaml解析器會把不帶雙引號的 123456 解析成 int。
加載配置文件
我們有了配置定義(config.go)和配置文件(config.yaml),接下來就是加載配置文件了,加載配置文件有三種方式:
- 必須加載成功,否則程序退出,我們一般這么用,如果配置不對,程序就無法繼續(xù)了
// 有錯誤直接退出程序
var config RestfulConf
conf.MustLoad("config.yaml", &config)
go-zero 自帶的 goctl 生成的默認代碼也是使用 MustLoad 來加載配置文件的
- 加載配置,并自行判斷是否有 error
// 自己判斷并處理 error
var config RestfulConf
// 為了更簡潔,這里的 LoadConfig 后續(xù)會改為 Load,LoadConfig 已被標(biāo)記為 Deprecated
if err := conf.LoadConfig("config.yaml", &config); err != nil {
log.Fatal(err)
}
- 加載配置并讀取環(huán)境變量
// 自動讀取環(huán)境變量 var config RestfulConf conf.MustLoad(configFile, &config, conf.UseEnv())
這里為啥我們需要顯式指定 conf.UseEnv(),因為如果默認讀取的話,可能在配置里大家寫特定字符的時候就需要 escape 了,所以默認不讀取環(huán)境變量,這個設(shè)計也歡迎大家多提提建議哈
實現(xiàn)原理
我們在實現(xiàn)類似 yaml/json 解析的時候一般會直接使用 encoding/json 或者對應(yīng)的 yaml 庫,但是對于 go-zero 來說,我們需要在 unmarshal 的時候有更精確的控制,這就需要我們自己定制 yaml/json 的解析了,完整的代碼實現(xiàn)在:
配置文件代碼:github.com/zeromicro/g…
yaml/json 解析代碼:github.com/zeromicro/g…
這里也充分展示了 reflect 的用法,以及復(fù)雜場景下如何通過單元測試保證代碼的正確性。
總結(jié)
我一直比較推薦 Fail Fast 的思想,我們在加載配置文件的時候也是這樣,一旦有錯誤,立馬退出,這樣運維在部署服務(wù)時就會及時發(fā)現(xiàn)問題,因為進程壓根起不來。
go-zero 的所有服務(wù)的配置項都是通過這樣的方式來加載和自動驗證的,包括我寫的很多工具的配置也是基于此來實現(xiàn)的
更多關(guān)于Go配置文件定義讀取的資料請關(guān)注腳本之家其它相關(guān)文章!希望能對你有所幫助!
相關(guān)文章
golang實現(xiàn)讀取excel數(shù)據(jù)并導(dǎo)入數(shù)據(jù)庫
Go 語言是一門適合用于編寫高效且并發(fā)的 Web 應(yīng)用程序的編程語言,同時也可以使用它進行數(shù)據(jù)處理和分析,本文主要介紹了如何通過go語言實現(xiàn)讀取excel數(shù)據(jù)并導(dǎo)入數(shù)據(jù)庫,感興趣的小伙伴可以了解下2025-04-04
利用ChatGPT編寫一個Golang圖像壓縮函數(shù)
這篇文章主要為大家詳細介紹了如何利用ChatGPT幫我們寫了一個Golang圖像壓縮函數(shù),文中的示例代碼簡潔易懂,感興趣的小伙伴可以嘗試一下2023-04-04
Golang設(shè)計模式工廠模式實戰(zhàn)寫法示例詳解
這篇文章主要為大家介紹了Golang 工廠模式實戰(zhàn)寫法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-08-08
golang結(jié)構(gòu)化日志slog的用法簡介
日志是任何軟件的重要組成部分,Go?提供了一個內(nèi)置日志包(slog),在本文中,小編將簡單介紹一下slog包的功能以及如何在?Go?應(yīng)用程序中使用它,感興趣的可以了解下2023-09-09

