当前位置: 首页 > news >正文

【Go】Go viper 配置模块

1. 配置相关概念

在项目开发过程中,一旦涉及到与第三方中间件打交道就不可避免的需要填写一些配置信息,例如 MySQL 的连接信息、Redis 的连接信息。如果这些配置都采用硬编码的方式无疑是一种不优雅的做法,有以下缺陷:

  • 不同环境(开发环境、测试环境、线上环境)之间无法做到环境隔离
  • 信息分散,无法做到集中统一管理
  • 与业务代码相耦合,无法灵活调整

因此配置模块应运而生,这也是为什么在 Go 项目中需要引入 viper 配置模块的原因!

1.1 配置的来源与优先级

在项目中我们可以从不同地方读取配置,常见的配置信息的来源有如下几类:

  1. 命令行参数:往往是启动项目时读取的,比如 mockgen 命令行有 source、destination、package等等参数
  2. 环境变量:配置信息也可以从操作系统当中设置的环境变量中读取
  3. 本地配置文件:比如在 Java 项目中,常常会用 application.yaml 类似的格式保存配置信息
  4. 远程配置中心:这是一个独立的服务,比如 nacos、etcd 就是常见的配置中心

对于上面常见的四种配置来源,如果相互之间冲突会有什么影响呢?这就涉及到配置来源的优先级了,一般来说优先级如下:命令行启动参数 > 本地配置文件 -> 系统环境变量 -> 远程配置中心

💡 温馨提示:上述优先级排列仅个人观点,不同公司有不同的规范标准!

2. viper 简介

viper 是一个 Go 项目当中的配置框架,在项目中引入 viper 可以轻松的读取配置信息当中设定的参数并在项目中加以使用

⭐ viper项目地址:https://github.com/spf13/viper

3. viper 快速入门

3.1 viper 读取配置文件

首先我们需要先在项目中编写配置文件,内容如下:

db:mysql:# mysql dsn连接字符串dsn: root:QWEzxc123456@tcp(localhost:3306)/webook
redis:# redis 连接地址addr: localhost:6379

该配置文件对应的存储路径位于config/dev.yaml

step1:首先我们需要设定好读取的配置文件的名称以及文件后缀,对应 API 如下:

  • AddConfigPath:设定配置文件所在跟路径
  • SetConfigName:设定配置文件名称
  • SetConfigType:设定配置文件类型
  • SetConfigFile:设定配置文件路径(可以替换上述三个参数)
  • SetDefault:设定默认值
  • ReadInConfig:读取配置文件
func InitViperLocal() {// 1. 设置配置文件路径viper.AddConfigPath("config")// 2. 设置配置文件名称viper.SetConfigName("dev")// 3.设置配置文件类型viper.SetConfigType("yaml")// 4. 进行读取err := viper.ReadInConfig()if err != nil {panic(err)}
}

step2:然后我们就可以尝试在项目中进行读取,对应 API 如下:

  • GetString:读取指定 key 对应的字符串数据
  • UnmarshalKey:读取指定 key 对应的数据到反序列化到指定结构体当中
func main() {InitViperLocal()// 1. 读取字符串数据dsn := viper.GetString("db.mysql.dsn")fmt.Println(dsn)// 2. 读取数据反序列到结构体当中var dbConfig DBConfigerr := viper.UnmarshalKey("db.mysql", &dbConfig)if err != nil {panic(err)}fmt.Println(dbConfig)
}

程序运行结果:

💡 温馨提示:如果在 GoLand 中无法读取配置文件路径,有可能是因此 Working Directory 工作目录的设置错误,在 IDE 做相应调整即可!

3.2 viper 读取启动参数

viper 还有一个特别好用的特性:可以通过读取命令行参数实现不同环境配置文件的隔离,比如在此项目中我们设定的配置文件路径为config/dev.yaml,因此我们在命令行参数中指定为--config=config/dev.yaml,如果想要设定测试环境就可以指定命令行参数为:--config=config/test.yaml,因此只需要使用 viper 读取命令行参数来设定配置文件路径即可!

step1:在 Goland 中设定运行时的启动命令行参数

step2:首先我们需要读取命令行当中的参数,对应 API 如下:

  • pflag.String:设置从命令行中读取的参数名称、默认值、说明
  • pflag.Parse:解析读取到的命令行参数
func InitViperCLI() {// 1. 读取命令行参数s := pflag.String("config", "", "配置文件路径")// 2. 解析参数pflag.Parse()// 3. 设定配置文件路径viper.SetConfigFile(*s)// 4. 读取配置err := viper.ReadInConfig()if err != nil {panic(err)}
}

step3:然后我们就可以尝试在项目中正常读取

func main() {//InitViperLocal()InitViperCLI()// 1. 读取字符串数据addr := viper.GetString("redis.addr")fmt.Println(addr)
}

程序运行结果:

3.3 viper 读取配置中心

3.3.1 安装 etcd

step1:使用 Docker 安装 etcd,docker-compose.yaml 文件内容如下:

version: "3"
services:# 配置etcdetcd:image: "bitnami/etcd:latest"restart: alwaysentrypoint:- ALLOW_NON_AUTHENTICATION=yesports:- "12379:2379"

然后在同级目录运行docker compose up启动:

step2:安装 etcdctl 工具,下载源码包:git clone [https://github.com/etcd-io/etcd.git](https://github.com/etcd-io/etcd.git),切换到正式版本tag中,进入etcdctl目录执行go install .命令

然后就会在GOPATH/bin目录下生成对应工具,运行etcdctl命令进行验证

3.3.2 使用 viper 接入 etcd

step1:使用 etcdctl 工具存储配置文件:etcdctl --endpoints=127.0.0.1:2379 put /viper_demo "$(<dev.yaml)"

step2:在项目中通过以下 API 读取远程配置:

  • AddRemoteProvider:设置远程配置中心名称、地址、路径
  • ReadRemoteConfig:读取远程配置
func InitViperRemote() {// 1. 连接远程配置中心err := viper.AddRemoteProvider("etcd3", "127.0.0.1:2379", "/viper_demo")if err != nil {panic(err)}// 2. 读取配置err = viper.ReadRemoteConfig()if err != nil {panic(err)}
}

step3:在项目中匿名导入下面这个包:_ “github.com/spf13/viper/remote”

step4:在主函数中读取配置,验证结果:

func main() {//InitViperLocal()//InitViperCLI()InitViperRemote()// 1. 读取字符串数据addr := viper.GetString("redis.addr")fmt.Println(addr)
}

程序运行结果如下:

3.4 viper 监听配置变更

viper 提供了下列 API 用于监听配置文件的变更并执行对应的回调函数

  • WatchConfig:执行监听本地配置文件的变更
  • WatchRemoteConfig:执行监听远程配置中心的变更
  • OnConfigChange:当配置文件某个 Key 变更时执行对应的回调函数
func InitViperLocal() {// 1. 设置配置文件路径viper.AddConfigPath("config")// 2. 设置配置文件名称viper.SetConfigName("dev")// 3.设置配置文件类型viper.SetConfigType("yaml")// 4. 进行读取err := viper.ReadInConfig()if err != nil {panic(err)}// 5. 开启监听viper.WatchConfig()// 6. 设置变更回调viper.OnConfigChange(func(e fsnotify.Event) {fmt.Println("config file changed:", e.Name)fmt.Println(viper.GetString("db.mysql.dsn"))fmt.Println(viper.GetString("redis.addr"))})
}func main() {InitViperLocal()server := gin.Default()server.Run(":8080")
}

程序运行结果:

http://www.lryc.cn/news/546636.html

相关文章:

  • zabbix“专家坐诊”第277期问答
  • 大模型工程师学习日记(十一):FAISS 高效相似度搜索和密集向量聚类的库
  • python学习第三天
  • 深入解析 Svelte:下一代前端框架的革命
  • C++20 中位移位运算符的统一行为:深入解析与实践指南
  • Linux——基本指令
  • MySql面试总结(二)
  • Pytorch中的主要函数
  • Java实现大数据量导出报表
  • 大语言模型 智能助手——既能生成自然语言回复,又能在必要时调用外部工具获取实时数据
  • PyTorch 系统教程:理解机器学习数据分割
  • 分水岭算法(Watershed Algorithm)教程:硬币分割实例
  • 【STM32项目实战系列】基于STM32G474的FDCAN驱动配置
  • shell文本处理
  • 如何利用客户端双向TLS认证保护云上应用安全
  • nlp第十节——LLM相关
  • T-SQL 语言基础: SQL 数据库对象元数据及配置信息获取
  • ue5 创建多列StreeView的方法与理解
  • C# OnnxRuntime部署DAMO-YOLO香烟检测
  • 陕西省地标-DB61/T 1121-2018 政务服务中心建设和运营规范
  • UDP协议(20250303)
  • 【四.RAG技术与应用】【12.阿里云百炼应用(下):RAG的云端优化与扩展】
  • Docker新手入门(持续更新中)
  • 【星云 Orbit • STM32F4】08. 用判断数据头来接收据的串口通用程序框架
  • HSPF 水文模型建模方法与案例分析实践技术应用
  • 设置 CursorRules 规则
  • 人工智能AI在汽车设计领域的应用探索
  • 《当AI生成内容遭遇审核:需求与困境的深度剖析》:此文为AI自动生成
  • 【无人机与无人车协同避障】
  • ComfyUI AnimeDiff动画参数总结