Golang的viper库

1.作用

viper 是一个配置解决方案,拥有丰富的特性:

  • 支持 JSON/TOML/YAML/HCL/envfile/Java properties 等多种格式的配置文件;
  • 可以设置监听配置文件的修改,修改时自动加载新的配置;
  • 从环境变量、命令行选项和io.Reader中读取配置;
  • 从远程配置系统中读取和监听修改,如 etcd/Consul;
  • 代码逻辑中显示设置键值。

2.快速使用

安装:

$ go get github.com/spf13/viper

使用:

package mainimport ("fmt""log""github.com/spf13/viper"
)func main() {viper.SetConfigName("config")viper.SetConfigType("toml")viper.AddConfigPath(".")viper.SetDefault("redis.port", 6381)err := viper.ReadInConfig()if err != nil {log.Fatal("read config failed: %v", err)}fmt.Println(viper.Get("app_name"))fmt.Println(viper.Get("log_level"))fmt.Println("mysql ip: ", viper.Get("mysql.ip"))fmt.Println("mysql port: ", viper.Get("mysql.port"))fmt.Println("mysql user: ", viper.Get("mysql.user"))fmt.Println("mysql password: ", viper.Get("mysql.password"))fmt.Println("mysql database: ", viper.Get("mysql.database"))fmt.Println("redis ip: ", viper.Get("redis.ip"))fmt.Println("redis port: ", viper.Get("redis.port"))
}

config.toml

app_name = "awesome web"# possible values: DEBUG, INFO, WARNING, ERROR, FATAL
log_level = "DEBUG"[mysql]
ip = "127.0.0.1"
port = 3306
user = "dj"
password = 123456
database = "awesome"[redis]
ip = "127.0.0.1"
port = 7381

viper 的使用非常简单,它需要很少的设置。设置文件名(SetConfigName)、配置类型(SetConfigType)和搜索路径(AddConfigPath),然后调用ReadInConfig
viper会自动根据类型来读取配置。使用时调用viper.Get方法获取键值。

编译、运行程序:

awesome web
DEBUG
mysql ip:  127.0.0.1
mysql port:  3306
mysql user:  dj
mysql password:  123456
mysql database:  awesome
redis ip:  127.0.0.1
redis port:  7381

有几点需要注意:

  • 设置文件名时不要带后缀;
  • 搜索路径可以设置多个,viper 会根据设置顺序依次查找;
  • viper 获取值时使用section.key的形式,即传入嵌套的键名;
  • 默认值可以调用viper.SetDefault设置。

3.读取键

viper 提供了多种形式的读取方法。在上面的例子中,我们看到了Get方法的用法。Get方法返回一个interface{}的值,使用有所不便。

GetType系列方法可以返回指定类型的值。
其中,Type 可以为Bool/Float64/Int/String/Time/Duration/IntSlice/StringSlice
但是请注意,如果指定的键不存在或类型不正确,GetType方法返回对应类型的零值

如果要判断某个键是否存在,使用IsSet方法。
另外,GetStringMapGetStringMapString直接以 map 返回某个键下面所有的键值对,前者返回map[string]interface{},后者返回map[string]string
AllSettingsmap[string]interface{}返回所有设置。

// 省略包名和 import 部分func main() {viper.SetConfigName("config")viper.SetConfigType("toml")viper.AddConfigPath(".")err := viper.ReadInConfig()if err != nil {log.Fatal("read config failed: %v", err)}fmt.Println("protocols: ", viper.GetStringSlice("server.protocols"))fmt.Println("ports: ", viper.GetIntSlice("server.ports"))fmt.Println("timeout: ", viper.GetDuration("server.timeout"))fmt.Println("mysql ip: ", viper.GetString("mysql.ip"))fmt.Println("mysql port: ", viper.GetInt("mysql.port"))if viper.IsSet("redis.port") {fmt.Println("redis.port is set")} else {fmt.Println("redis.port is not set")}fmt.Println("mysql settings: ", viper.GetStringMap("mysql"))fmt.Println("redis settings: ", viper.GetStringMap("redis"))fmt.Println("all settings: ", viper.AllSettings())
}

我们在配置文件 config.toml 中添加protocolsports配置:

[server]
protocols = ["http", "https", "port"]
ports = [10000, 10001, 10002]
timeout = 3s

编译、运行程序,输出:

protocols:  [http https port]
ports:  [10000 10001 10002]
timeout:  3s
mysql ip:  127.0.0.1
mysql port:  3306
redis.port is set
mysql settings:  map[database:awesome ip:127.0.0.1 password:123456 port:3306 user:dj]
redis settings:  map[ip:127.0.0.1 port:7381]
all settings:  map[app_name:awesome web log_level:DEBUG mysql:map[database:awesome ip:127.0.0.1 password:123456 port:3306

4.设置键值

viper 支持在多个地方设置,使用下面的顺序依次读取:

  • 调用Set显示设置的;
  • 命令行选项;
  • 环境变量;
  • 配置文件;
  • 默认值。

viper.Set

如果某个键通过viper.Set设置了值,那么这个值的优先级最高。

viper.Set("redis.port", 5381)

如果将上面这行代码放到程序中,运行程序,输出的redis.port将是 5381。

命令行选项

如果一个键没有通过viper.Set显示设置值,那么获取时将尝试从命令行选项中读取。
如果有,优先使用。viper 使用 pflag 库来解析选项。
我们首先在init方法中定义选项,并且调用viper.BindPFlags绑定选项到配置中:

func init() {pflag.Int("redis.port", 8381, "Redis port to connect")// 绑定命令行viper.BindPFlags(pflag.CommandLine)
}

然后,在main方法开头处调用pflag.Parse解析选项。

编译、运行程序:

$ ./main.exe --redis.port 9381
awesome web
DEBUG
mysql ip:  127.0.0.1
mysql port:  3306
mysql user:  dj
mysql password:  123456
mysql database:  awesome
redis ip:  127.0.0.1
redis port:  9381

如何不传入选项:

$ ./main.exe
awesome web
DEBUG
mysql ip:  127.0.0.1
mysql port:  3306
mysql user:  dj
mysql password:  123456
mysql database:  awesome
redis ip:  127.0.0.1
redis port:  7381

注意,这里并不会使用选项redis.port的默认值。

但是,如果通过下面的方法都无法获得键值,那么返回选项默认值(如果有)。试试注释掉配置文件中redis.port看看效果。

环境变量

如果前面都没有获取到键值,将尝试从环境变量中读取。我们既可以一个个绑定,也可以自动全部绑定。

init方法中调用AutomaticEnv方法绑定全部环境变量:

func init() {// 绑定环境变量viper.AutomaticEnv()
}

为了验证是否绑定成功,我们在main方法中将环境变量 GOPATH 打印出来:

func main() {// 省略部分代码fmt.Println("GOPATH: ", viper.Get("GOPATH"))
}

通过 系统 -> 高级设置 -> 新建 创建一个名为redis.port的环境变量,值为 10381。
运行程序,输出的redis.port值为 10381,并且输出中有 GOPATH 信息。

也可以单独绑定环境变量:

func init() {// 绑定环境变量viper.BindEnv("redis.port")viper.BindEnv("go.path", "GOPATH")
}func main() {// 省略部分代码fmt.Println("go path: ", viper.Get("go.path"))
}

调用BindEnv方法,如果只传入一个参数,则这个参数既表示键名,又表示环境变量名。
如果传入两个参数,则第一个参数表示键名,第二个参数表示环境变量名。

还可以通过viper.SetEnvPrefix方法设置环境变量前缀,这样一来,通过AutomaticEnv和一个参数的BindEnv绑定的环境变量,
在使用Get的时候,viper 会自动加上这个前缀再从环境变量中查找。

如果对应的环境变量不存在,viper 会自动将键名全部转为大写再查找一次。所以,使用键名gopath也能读取环境变量GOPATH的值。


5.参考文章

传送门

Golang的viper库相关推荐

  1. 再测Golang的JSON库

    2019独角兽企业重金招聘Python工程师标准>>> 写项目一直需要进行序列化,听到了,也看到了很多同学老师对各个golang的json库进行测评.那本人为什么还要继续进行这一次测 ...

  2. Golang实现requests库

    Golang实现requests库 简单的封装下,方便使用,像python的requests库一样. Github地址 Github 支持 GET.POST.PUT.DELETE applicatio ...

  3. go语言调用c 的头文件 so,Golang生成共享库(shared library)以及Golang生成C可调用的动态库.so和静态库.a...

    Golang类似于C的静态语言,效率也接近于C,如果Golang也可以导出可供C调用的库,那可以和很多高级语言say goodbye了,goodbye似乎又有点武断,但至少说,Golang可以做很多事 ...

  4. 用 Golang 实现矩阵运算库

    用 Golang 实现矩阵运算库 天下文章一大抄 抄过来的 转载的 别来沾边

  5. GoLang之标准库net/http包源码

    GoLang之标准库net/http包源码 目录 GoLang之标准库net/http包源码 1.net/http包介绍 2.http.ListenAndServe函数 3.http.Handle函数 ...

  6. Golang高性能日志库zap + lumberjack 日志切割组件详解

    文章篇幅较长,可以先收藏防止迷路~ 目录 zap日志库 1. why zap? 2. 简单使用 3. 自定义logger例子 4. Gin项目使用zap 6. lumberjack 日志切割组件 za ...

  7. golang桌面gui库fyne使用

    golang桌面gui库fyne使用 使用 初始项目 新建项目文件夹fyna 执行go mod init 执行go get fyne.io/fyne/v2 执行go mod tidy 创建main.g ...

  8. GoLang之标准库strings包

    文章目录 GoLang之标准库strings包 1.函数介绍 2.EqualFold 3.TrimSpace 4.HasPrefix 5.HasSuffix 6.Index 7.LastIndex 8 ...

  9. golang高性能日志库zap的使用

    本文作者:陈进坚 个人博客:https://jian1098.github.io CSDN博客:https://blog.csdn.net/c_jian 简书:https://www.jianshu. ...

最新文章

  1. 深入了解line-height
  2. c语言数组两个值交换,如可交换两个数组中的元素?
  3. Java ArrayList contains()方法及示例
  4. SVN:通过命令行只提交目录,并忽略其中的文件
  5. [特约· Keso ·东拉西扯] 我眼中的乔布斯
  6. java获取当月1号 的时间chuo_java获取时间戳的方法
  7. 数值分析(8)-最佳一致逼近多项式
  8. 费尔个人防火墙采用两种封包过滤技术
  9. php 车牌号 正则,php正则验证车牌格式的函数
  10. 无法打开包括文件: “type_traits”
  11. 怎么配置内网IP SSL证书?
  12. 北京林业大学matlab公选课,北京林业大学视频类公共选修课学习指引-北京林业大学教务处.DOC...
  13. 论文笔记(十八):Object Detection and Spatial Location Method for ... Based on 3D Virtual Geographical Scen
  14. 压缩包解压后的文件名是乱码怎么解决
  15. 如何用计算机环境设计,计算机绘图软件在环境艺术设计中运用.doc
  16. 臭可乐,怎么才能让你不吃便便呢,唉
  17. [zz]浅谈自然语言处理(NLP)和 自然语言理解(NLU)
  18. ECIF OCRM ACRM
  19. 印度IT业迎来新生:大数据催生大批分析公司
  20. 联想G455 XP/MAC 双系统安装

热门文章

  1. 浙江大学计算机学院 潘刚,潘刚 - 江苏科技大学 - 生物技术学院
  2. 求解随机规划的情景树,情景规划 scenario 方法
  3. CD-ICBC-常见问题或开发过程备份记录
  4. C. Chocolate Bunny(思维+规律)
  5. 阿里巴巴微服务架构演进
  6. 好心情:长期服用精神药物,需监测哪些指标?
  7. XML Publisher 模板
  8. 盘点激光雷达常用的10大数据处理软件
  9. 京东金融的贷款规模创新高
  10. Copernicus Open Access Hub哥白尼开放中心