欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

淺談Golang如何使用Viper進行配置管理

 更新時間:2023年06月27日 11:44:03   作者:Shawn27  
在Golang生態(tài)中,Viper是一個不錯的開源配置管理框架,這篇文章主要為大家介紹了Golang如何使用Viper進行配置管理,需要的可以參考一下

前言

應用程序里要用到很多配置,比如監(jiān)聽端口、數(shù)據(jù)庫相關配置、緩存相關配置等。這些配置可能來自不同的源,最常見的是本地配置文件,也可以作為命令行參數(shù)或環(huán)境變量傳入,還可以托管在遠程配置中心。同一項配置可能在多個不同的配置源中同時存在,如何處理優(yōu)先級?在程序運行過程中,有些配置需要能夠動態(tài)更新,又該如何實現(xiàn)?配置的源、優(yōu)先級及動態(tài)更新,就是配置管理的內容。

Golang生態(tài)中,Viper是一個不錯的開源配置管理框架。

配置層級

Viper支持多種配置源,并處理了它們之間的優(yōu)先級問題,按優(yōu)先級從高到低的順序,層級如下:

  • 命令行參數(shù)
  • 環(huán)境變量
  • 本地配置文件
  • 遠程配置文件
  • 默認值

下面分別看一下Viper是如何支持這些配置源的。

0. 配置結構

以如下的yaml格式配置文件來說明:

# myconfig.yaml
aaa: foo
xxx:
    bbb: bar
    ccc: 100

Viper在內存中是用map類型來維護配置的,以配置名作為key、配置值作為Value。

為方便使用,我們可以定義如下的結構體來接收配置:

package config
var Config config
type config struct {
    Aaa string
    Xxx struct {
        Bbb string
        Ccc uint
    }
}

1. 命令行參數(shù)

Viper提供了BindPFlags函數(shù)來綁定命令行參數(shù),以Cobra命令行框架為例,初始化代碼如下:

// 1. 根據(jù)需要定義命令行參數(shù)
pflags = rootCmd.PersistentFlags()
pflags.StringP("aaa", "a", "", "aaa's usage")
pflags.String("xxx.bbb", "", "xxx.bbb's usage")
pflags.Uint("xxx.ccc", 0, "xxx.ccc's usage")
// 2. Viper綁定命令行參數(shù)
viper.BindPFlags(pflags)

這樣就可以通過命令行參數(shù)來傳入相關配置:

// 參數(shù)aaa使用長參數(shù)格式
$ mycmd --aaa foo --xxx.bbb bar --xxx.ccc 100
// 參數(shù)aaa使用短參數(shù)格式
$ mycmd -a foo --xxx.bbb bar --xxx.ccc 100

2. 環(huán)境變量

Viper提供了AutomaticEnv函數(shù)來從環(huán)境變量中加載配置,示例代碼如下:

// 設置環(huán)境變量的前綴
viper.SetEnvPrefix("MYCMD")
// 將配置Key中的點號(.)和橫杠(-)替換為下劃線(_)
viper.SetEnvKeyReplacer(strings.NewReplacer(".", "_", "-", "_"))
// 從環(huán)境變量中讀取匹配的配置
viper.AutomaticEnv()

這樣就可以通過環(huán)境變量來傳入相關配置:

$ MYCMD_AAA=foo MYCMD_XXX_BBB=bar MYCMD_XXX_CCC=100 mycmd

3. 本地配置文件

Viper提供了從本地配置文件加載配置的方法,示例代碼如下:

// 在初始化cobra時,將本地配置文件路徑定義為命令行參數(shù)
pflags.StringVarP(&cfgFile, "config", "c", "", "config file")
// 在初始化viper時,設置本地配置文件路徑
viper.SetConfigFile(cfgFile)
// 讀取本地配置文件
if err := viper.ReadInConfig(); err != nil {
    log.Fatal(err)
}

這樣就可以通過本地配置文件來傳入相關配置:

// myconf.yaml為同目錄下的配置文件,內容如上文所述
$ mycmd --config myconf.yaml

4. 遠程配置文件

Viper還支持將配置文件托管在遠程配置中心,如etcd,示例代碼如下:

// 注意:在實際項目中,以下參數(shù)也可以作為配置傳入
provider := "etcd3"
endpoint := "x.x.x.x:2379"
path := "/config/mycmd/myconf.yaml"
viper.AddRemoteProvider(provider, endpoint, path)
viper.SetConfigType("yaml")
if err := viper.ReadRemoteConfig(); err != nil {
    log.Println("Read remote config failed:", err)
}

只要將myconf.yaml配置文件上傳到etcd的指定路徑下,程序就可以遠程讀取了。

$ cat myconf.yaml | etcdctl put /config/mycmd/myconf.yaml

5. 默認值

Viper還支持設置默認值,對于在以上所有源中都不存在的配置,將使用其默認值,示例代碼如下:

viper.SetDefault("aaa", "foo")
viper.SetDefault("xxx.bbb", "bar")
viper.SetDefault("xxx.ccc", 100)

在所有配置源都加載后,我們可以將Viper配置裝載到配置結構體變量中,方便使用:

if err := viper.Unmarshal(&config.Config); err != nil {
    log.Fatal("Viper unmarshal failed:", err)
}

配置動態(tài)更新

以上解決了靜態(tài)配置管理的問題,如何實現(xiàn)配置的動態(tài)更新呢?

配置的動態(tài)更新包括三個步驟:

  • 更新配置源中的配置
  • 更新內存中的配置
  • 配置的動態(tài)生效

注意:如Viper官方文檔所提,Viper本身不是并發(fā)安全的,在實現(xiàn)配置動態(tài)更新時,要注意采用鎖機制等方式來保證Viper并發(fā)讀寫的安全。

1. 更新配置源中的配置

Viper支持的配置源中,命令行參數(shù)、環(huán)境變量是在進程啟動時一次性讀取的,不支持動態(tài)更新。本地配置文件和遠程配置文件可以支持動態(tài)更新,直接修改配置文件即可。

2. 更新內存中的配置

Viper支持監(jiān)聽本地配置文件的變化,同時允許注冊文件變化時的回調函數(shù),借助這個功能可以實現(xiàn)本地配置文件的動態(tài)更新,示意代碼如下:

viper.OnConfigChange(func(e fsnotify.Event) {
    // 回調函數(shù)
    // 可在此處更新內存中的本地配置:
    // 1. 重新讀取本地配置:ReadInConfig
    // 2. 裝載到配置結構體:Unmarshal
    // 注意:為Viper及配置結構體加鎖
})
viper.WatchConfig()

對于遠程配置文件,可以通過周期性地重新讀取來實現(xiàn)動態(tài)更新,示意代碼如下:

go func() {
    for {
        time.Sleep(time.Second * 10)
        // 周期為10s
        // 可在此處更新內存中的遠程配置:
        // 1. 重新讀取遠程配置:WatchRemoteConfig
        // 2. 裝載到配置結構體:Unmarshal
        // 注意:為Viper及配置結構體加鎖
    }
}()

3. 配置的動態(tài)生效

不同配置的使用方式不同,動態(tài)生效方式也就不同,大概可以分為兩類:

  • 有些配置,每次使用時都是讀取最新的值,因此只要更新了內存中的配置,就會即時生效
  • 有些配置,比如數(shù)據(jù)庫憑據(jù),可能需要重建連接池才能生效

在這里就不再細說了。

總結

本文介紹了在Golang中優(yōu)雅、靈活地實現(xiàn)配置管理的一種方式,涵蓋配置的源、優(yōu)先級及動態(tài)更新等內容。重點說明了命令行參數(shù)、環(huán)境變量、本地配置文件、遠程配置文件等多種配置源的實現(xiàn)方法,以及配置動態(tài)更新的實現(xiàn)思路。

以上就是淺談Golang如何使用Viper進行配置管理的詳細內容,更多關于Go Viper配置管理的資料請關注腳本之家其它相關文章!

相關文章

  • Golang源碼分析之golang/sync之singleflight

    Golang源碼分析之golang/sync之singleflight

    golang/sync庫拓展了官方自帶的sync庫,提供了errgroup、semaphore、singleflight及syncmap四個包,本次先分析第一個包errgroup的源代碼,下面這篇文章主要給大家介紹了關于Golang源碼分析之golang/sync之singleflight的相關資料,需要的朋友可以參考下
    2022-11-11
  • golang實現(xiàn)延遲隊列(delay queue)的兩種實現(xiàn)

    golang實現(xiàn)延遲隊列(delay queue)的兩種實現(xiàn)

    本文主要介紹了golang實現(xiàn)延遲隊列(delay queue)的兩種實現(xiàn),文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2025-05-05
  • Go調用opencv實現(xiàn)圖片矯正的代碼示例

    Go調用opencv實現(xiàn)圖片矯正的代碼示例

    這篇文章主要為大家詳細介紹了Go調用opencv實現(xiàn)圖片矯正的代碼示例,文中的示例代碼簡潔易懂,感興趣的小伙伴可以跟隨小編一起學習一下
    2023-09-09
  • golang使用mTLS實現(xiàn)雙向加密認證http通信

    golang使用mTLS實現(xiàn)雙向加密認證http通信

    這篇文章主要為大家介紹了golang如何調用mTLS實現(xiàn)雙向加密認證http通信,文中的示例代碼講解詳細,具有一定的學習價值,需要的小伙伴可以參考下
    2023-08-08
  • 淺析在Go語言中如何實現(xiàn)協(xié)程池

    淺析在Go語言中如何實現(xiàn)協(xié)程池

    gammazero/workerpool?就是用來實現(xiàn)協(xié)程池的?Go?包,本文我們將一起來學習一下其使用方法,并深入其源碼來探究下如何實現(xiàn)一個?Go?協(xié)程池,需要的可以了解下
    2025-06-06
  • 深入Golang之context的用法詳解

    深入Golang之context的用法詳解

    本篇文章主要介紹了深入Golang之context的用法詳解,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧
    2018-03-03
  • B站新一代 golang規(guī)則引擎gengine基礎語法

    B站新一代 golang規(guī)則引擎gengine基礎語法

    這篇文章主要為大家介紹了B站新一代 golang規(guī)則引擎gengine基礎語法,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2023-12-12
  • Go語言struct類型詳解

    Go語言struct類型詳解

    這篇文章主要介紹了Go語言struct類型詳解,struct是一種數(shù)據(jù)類型,可以用來定義自己想的數(shù)據(jù)類型,需要的朋友可以參考下
    2014-10-10
  • golang?run時報undefined錯誤的解決

    golang?run時報undefined錯誤的解決

    這篇文章主要介紹了golang?run時報undefined錯誤的解決方案,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2024-03-03
  • Go語言如何判斷兩張圖片的相似度

    Go語言如何判斷兩張圖片的相似度

    這篇文章主要為大家詳細介紹了Go語言如何中實現(xiàn)判斷兩張圖片的相似度的兩種方法,文中的示例代碼講解詳細,感興趣的小伙伴可以跟隨小編一起學習一下
    2025-06-06

最新評論