【viper】go 配置管理神器viper使用详解

news2025/1/20 3:35:26

目录

  • 一、viper简介
  • 二、基本使用
    • 1.viper基本配置
    • 2.读取配置
    • 3.自定义配置源
    • 4.注册别名
    • 4.获取环境变量
    • 5.获取命令行参数
    • 6.多实例使用
    • 7.监听配置变化
  • 三、读取远程配置
  • 四、保存配置

一、viper简介

viper是一个应用程序解决方案,它支持在应用程序内使用,特性如下:

  • 支持json、toml、yaml、yml、properties、props、prop、hcl、tfvars、dotenv、env、ini 等格式的配置文件;
  • 支持从文件、环境变量、命令行、io.Reader、远程配置中心(etcd、etcd3、consul、firestore) 读取和修改配置;
  • 可以监听配置文件修改,并应用到程序中。

二、基本使用

1.viper基本配置

  • func (v *Viper) SetConfigType(in string) // 设置配置文件格式
  • func (v *Viper) SetConfigName(in string) // 设置配置文件名
  • func (v *Viper) AddConfigPath(in string) // 添加配置文件寻找路径
  • func (v *Viper) ReadInConfig() // 读取配置
func main(){
	viper.SetConfigType("yaml")        
	viper.SetConfigName("config.yaml") 
	viper.AddConfigPath(".")           
	// 读取配置
	if err := viper.ReadInConfig(); err != nil {
		if _, ok := err.(viper.ConfigFileNotFoundError); ok {
			fmt.Println("file not found.")
			return
		}
		panic(err)
	}
}

2.读取配置

viper 支持读取不同类型的数据。

  • func (v *Viper)GetString(key string) string
  • func (v *Viper)GetBool(key string) bool
  • func (v *Viper)GetInt(key string) int
  • func (v *Viper)GetInt32(key string) int32
  • func (v *Viper)GetInt64(key string) int64
  • func (v *Viper)GetUint(key string) uint
  • func (v *Viper)GetUint16(key string) uint16
  • func (v *Viper)GetUint32(key string) uint32
  • func (v *Viper)GetUint64(key string) uint64
  • func (v *Viper)GetFloat64(key string) float64
  • func (v *Viper)GetTime(key string) time.Time
  • func (v *Viper)GetDuration(key string) time.Duration
  • func (v *Viper)GetIntSlice(key string) []int
  • func (v *Viper)GetStringSlice(key string) []string
  • func (v *Viper)GetStringMap(key string) map[string]interface{}
  • func (v *Viper)GetStringMapString(key string) map[string]string
  • func (v *Viper)GetStringMapStringSlice(key string) map[string][]string
  • func (v *Viper)AllSettings() map[string]interface{} // 获取所有的配置项
  • func (v *Viper)IsSet(key string) bool //配置是否存在
    例如我有个配置文件为 config.yaml
# file : config.yaml
type: yaml
a1:
    k1: v1
    k2: v2
array:
    - 1
    - 2
    - 3
a1.k1: vv1

代码为:

func main(){
	viper.SetConfigType("yaml")
	viper.SetConfigName("config.yaml")
	viper.AddConfigPath(".")
	if err := viper.ReadInConfig(); err != nil {
		if _, ok := err.(viper.ConfigFileNotFoundError); ok {
			fmt.Println("file not found.")
			return
		}
		panic(err)
	}
	fmt.Println("type:", viper.GetString("type"))         // 读取字符串 type
	fmt.Println("a1:", viper.GetStringMap("a1"))          // 读取map a1
	fmt.Println("a1.k1:", viper.GetStringMap("a1")["k1"]) // 读取字符串a1.k1
	fmt.Println("a1.k1:", viper.GetString("a1.k1"))       // 嵌套读取字符串 a1.k1 注意和上面的区分,优先读取配置中和key完整匹配的部分
	fmt.Println("array:", viper.GetIntSlice("array"))     // 读取数组 array
}	

执行结果:
在这里插入图片描述
注意:获取配置的的方法,在key不存在的情况下,获取到的是0或者空值,可以先用 IsSet判断,例如:

if viper.IsSet("key") {
	fmt.Println("get key:", viper.GetString("key"))
}

3.自定义配置源

  • func (v *Viper) ReadConfig(in io.Reader) // 使用 io.Reader 作为配置输入源
func main(){
	yamlConfig := `
type: yaml
a1:
    k1: v1
    k2: v2
array:
    - 1
    - 2
    - 3
a1.k1: vv1
`
	viper.SetConfigType("yaml")
	
	if err := viper.ReadConfig(strings.NewReader(yamlConfig)); err != nil {
		panic(err)
	}
	fmt.Println(viper.AllSettings())
}	

4.注册别名

  • func (v *Viper)RegisterAlias(alias string, key string) // 注册别名
func main(){
	viper.SetConfigType("yaml")        // 设置配置文件格式
	viper.SetConfigName("config.yaml") // 设置配置文件名
	viper.AddConfigPath(".")           // 添加配置文件寻找路径
	// 读取配置
	if err := viper.ReadInConfig(); err != nil {
		if _, ok := err.(viper.ConfigFileNotFoundError); ok {
			fmt.Println("file not found.")
			return
		}
		panic(err)
	}

	viper.RegisterAlias("t", "type")
	fmt.Println(viper.GetString("t"))
}

这里可以看到,给 type 注册别名为 t,直接获取 t 就是 type 的配置。

4.获取环境变量

  • func (v *Viper)AutomaticEnv() // 自动绑定所有环境变量
  • func (v *Viper)AllowEmptyEnv(allowEmptyEnv bool) // 设置是否允许空环境变量
  • func (v *Viper)SetEnvPrefix(in string) // 设置前缀,用'_'和key连接
func main(){
	viper.AutomaticEnv()
	viper.AllowEmptyEnv(true)
	viper.RegisterAlias("env", "os_env")
	viper.SetEnvPrefix("viper")
	if err := os.Setenv("viper_os_env", "viper_alias_prefix_ok"); err != nil {
		panic(err)
	}
	fmt.Println(viper.GetString("env")) // 输出 viper_alias_prefix_ok
}

例子里可以看到,先给os_env命名别名为env,然后再加上前缀viper,所以最后获取环境变量传值为env时,实际获取的是viper_os_env

5.获取命令行参数

  • func (v *Viper)BindPFlag(key string, flag *pflag.Flag) error
  • func (v *Viper)BindPFlags(flags *pflag.FlagSet) error
    这里用到了 spf13 的 pflag 包,路径: github.com/spf13/pflag
func main(){
	pflag.IntP("int", "i", 1, "test int flag")
	pflag.StringP("string", "s", "s", "test string flag")
	pflag.Parse()

	e := viper.BindPFlag("string", pflag.Lookup("string"))
	if e != nil {
		panic(e)
	}
	fmt.Println(viper.GetString("string"))

	e = viper.BindPFlags(pflag.CommandLine)
	if e != nil {
		panic(e)
	}
	fmt.Println(viper.AllSettings())
}
// ./main.exe -t 2 -s ss
/* 结果
ss
map[int:2 string:ss]
*/

这里也可以使用大神的另一个包 cobra : github.com/spf13/cobra【后面补充个cobra的使用说明】

// 在init中绑定参数
func init() {
	cobra.OnInitialize(initCfg)
	rootCmd.PersistentFlags().StringP("version", "v", "6", "app version")
	viper.BindPFlag("version", rootCmd.PersistentFlags().Lookup("version"))
}

6.多实例使用

创建新的viper实例可以使用以下两种方法

  • func New() *Viper
  • func NewWithOptions(opts ...Option) *Viper // 根据配置创建实例
    其中第二种,viper 提供现有的配置有:
  • func KeyDelimiter(d string) Option // 设置配置项key的分隔符,默认是 '.' ,比如前面示例中的a1.k1
  • func EnvKeyReplacer(r StringReplacer) Option // 环境变量key的替换处理,如果使用Get方法获取环境变量时需要对key字符串进行某些替换动作可以传入处理接口
  • func IniLoadOptions(in ini.LoadOptions) Option // init类型配置文件加载选项
func testJson() {
	jsonViper := viper.New()
	jsonViper.SetConfigType("json")
	jsonViper.SetConfigName("config.json")
	jsonViper.AddConfigPath(".")
	if err := jsonViper.ReadInConfig(); err != nil {
		if _, ok := err.(viper.ConfigFileNotFoundError); ok {
			fmt.Println("file not found.")
			return
		}
		panic(err)
	}

	fmt.Println(jsonViper.AllSettings())
}

func testYaml() {
	yamlViper := viper.New()
	yamlViper.SetConfigType("yaml")       
	yamlViper.SetConfigName("config.yaml") 
	yamlViper.AddConfigPath(".")       

	if err := yamlViper.ReadInConfig(); err != nil {
		if _, ok := err.(viper.ConfigFileNotFoundError); ok {
			fmt.Println("file not found.")
			return
		}
		panic(err)
	}
	fmt.Println(yamlViper.AllSettings())
}


func main() {
	testJson()
	testYaml()
}

运行结果:
在这里插入图片描述

7.监听配置变化

  • func (v *Viper)OnConfigChange(run func(in fsnotify.Event)) // 注册配置变化处理函数
  • func (v *Viper)WatchConfig() // 开始监听配置变化
func main(){
	viper.SetConfigType("yaml")        // 设置配置文件格式
	viper.SetConfigName("config.yaml") // 设置配置文件名
	viper.AddConfigPath(".")           // 添加配置文件寻找路径
	// 读取配置
	if err := viper.ReadInConfig(); err != nil {
		if _, ok := err.(viper.ConfigFileNotFoundError); ok {
			fmt.Println("file not found.")
			return
		}
		panic(err)
	}
	viper.OnConfigChange(func(in fsnotify.Event) {
		fmt.Println("new type:", viper.GetString("type"))
	})
	viper.WatchConfig()
	time.Sleep(time.Minute)
}

三、读取远程配置

  • func (v *Viper) AddRemoteProvider(provider, endpoint, path string) // 新增远程配置
  • func (v *Viper) ReadRemoteConfig() error // 读取远程配置
  • func (v *Viper) WatchRemoteConfig() error // 监听远程配置,其实这个也是调用读取配置的接口,不是实际意义上的监听,需要循环调用
  • func (v *Viper) WatchRemoteConfigOnChannel() error // 通过 chan 监听配置,这个方法可以实时根据远程配置的变化修改本地数据,只需调用1次

这里以 etcd3 为例,不了解 etcd3 的同学可以看下我之前的博客。

etcd3 上现在已有配置
![在这里插入图片描述](https://img-blog.csdnimg.cn/3f8f142226cd45d8b4f190d5a67e1780.png

func main(){
	v := viper.New()
	v.SetConfigType("json")
	// 添加 etc3 远程配置中心
	err := v.AddRemoteProvider("etcd3", "http://10.1.30.79:12379", "/config/config.json")
	if err != nil {
		panic(err)
	}
	conf := &struct {
		Type string `json:"type"`
		A1   struct {
			K1 string `json:"k1"`
			K2 string `json:"k2"`
		} `json:"a1"`
		Array []int `json:"array"`
	}{}
	// 读取远程配置
	if err = v.ReadRemoteConfig(); err != nil {
		panic(err)
	}
	// 解析成json
	if err = v.Unmarshal(conf); err != nil {
		panic(err)
	}
	fmt.Println("conf:", conf)
	
	// 监听配置改变
	if err = v.WatchRemoteConfigOnChannel(); err != nil {
		panic(err)
	}
	
	// 循环打印所有配置项
	for {
		fmt.Println(v.AllSettings())
		time.Sleep(time.Second)
	}
}

四、保存配置

  • func (v *Viper) Set(key string, value interface{}) // 给配置项设置值
  • func (v *Viper) WriteConfig() error // 保存配置到文件中
  • func (v *Viper) SafeWriteConfig() error // 安全的保存配置到文件中,若文件已存在则报错
  • func (v *Viper) WriteConfigAs(filename string) error // 配置另存为文件
  • func (v *Viper) SafeWriteConfigAs(filename string) error // 安全的另存为配置文件,若文件已存在则报错
func main(){
	viper.SetConfigType("yaml")   // 设置配置文件格式
	viper.SetConfigName("config") // 设置配置文件名
	viper.AddConfigPath(".")      // 添加配置文件寻找路径
	// 读取配置
	if err := viper.ReadInConfig(); err != nil {
		if _, ok := err.(viper.ConfigFileNotFoundError); ok {
			fmt.Println("file not found.")
			return
		}
		panic(err)
	}
	
	viper.Set("type", "yml")
	if err := viper.SafeWriteConfig(); err != nil {
		panic(err)
	}
}

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/69338.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

基于元胞自动机的人员疏散模拟器研究(Matlab代码实现)

👨‍🎓个人主页:研学社的博客 💥💥💞💞欢迎来到本博客❤️❤️💥💥 🏆博主优势:🌞🌞🌞博客内容尽量做到思维缜…

[前端面试题]:数组去重的几种方法

总结一些日常需要用到的一些api,也是在一些面试中会经常出现的题目,今天分享的是数组去重的几个不同的方法, 同时文章也被收录到我的《JS基础》专栏中,欢迎大家点击收藏加关注。 给大家推荐一个实用面试题库 1、前端面试题库 &am…

魔兽世界开服架设教程—魔兽国服停服时间已确定!玩家该何去何从

《魔兽世界》国服停止运营时间定在,2023年1月24日0点起 近期随着暴雪和网易公司先后发出的公告,《魔兽世界》国服关服时间已经十分确定了,国服停运时间定在明年的1月24号0点起正式关服。随着国服将关闭,魔兽私服即将崛起。毕竟在游…

【Java面试】并发

文章目录线程有那些状态?一个程序来显示状态切换过程正常执行流程阻塞执行流程等待执行流程说说线程池的核心参数wait和sleep的区别Lock和synchronized的区别Lock中Condition的使用说说Java中的悲观锁与乐观锁乐观锁悲观锁Hashtable和ConcurrentHashMap的区别&#…

【玩转算法】(初始)动态规划

本期主题:动态规划,及其相关oj题。博客主页:小峰同学分享小编的在Linux中学习到的知识和遇到的问题小编的能力有限,出现错误希望大家不吝赐 🍁1)DP定义 动态规划是分治思想的延伸,通俗一点来说…

核酸检测预约系统毕业设计,核酸检测系统设计与实现,毕业设计怎么写论文毕设源码开题报告需求分析怎么做

项目背景和意义 目的:本课题主要目标是设计并能够实现一个基于web网页的疫情核酸检查预约系统,整个网站项目使用了B/S架构,基于java的springboot框架下开发;;通过后台设置医院信息、录入医院科室信息、录入医生信息、设…

CVE-2015-4852 Weblogic T3 反序列化分析

0x01 前言 看到很多师傅的面经里面都有提到 Weblogic 这一个漏洞,最近正好有一些闲暇时间,可以看一看。 因为环境上总是有一些小问题,所以会在本地和云服务器切换着调试 0x02 环境搭建 太坑了,我的建议是用本地搭建的方法&…

DeadObjectException解题

DeadObjectException解题 RemoteException occurs on reporting focusChanged, wWindow{2470935 u0 bundle_id/bundle_id.MainActivity}android.os.DeadObjectExceptionat android.os.BinderProxy.transactNative(Native Method)at android.os.BinderProxy.transact(Binder.ja…

十大开源测试工具和框架

免费的开源框架和工具由于其开源特性,现在逐渐成为自动化测试的首选解决方案。区别在于,你是喜欢使用类库编写一个全新的自动化测试框架,或者喜欢使用一个现成的工具。 本文帮忙你快速了解 10 大免费和开源测试工具、框架。 Katalon Studio…

【python】昔去雪如花,今来花似雪,今日份雪花快递到啦

前言 嗨喽~大家好呀,这里是魔王呐 ❤ ~ 大雪已至,冬天无恙,愿这个冬天的你,不缺暖阳,好日常在 小时候,冬天最盼望的就是下雪了~雪一下 在地上铺上一层雪白的毛毯的时候甚是好看~ 还能堆堆雪人、打打雪仗…

如何使用 django-import-export + pandas 在 Django 视图中导入 excel 数据

我想每个人都熟悉所有强大的库 django-import-export,它允许我们在 Django 的管理面板中导入和导出数据,但是如果你想让用户在网络上上传 excel 文件或通过 REST 上传它怎么办应用程序接口。最近,我遇到了类似的问题。经过大量研究,我找到了一种使用 Django 方式将数千个数…

基于jsp+mysql+ssm生鲜超市进销存管理系统-计算机毕业设计

项目介绍 目前超市越来越多,越来越普及,如何高效的管理经营超市才是成功的关键,其中对于中小型超市来说,在降低成本的前提下使用最有效的管理方式是非常重要的,所以开发中小型超市管理系统既考虑了成本相对较低又非常实用的特点。…

博客系统(前后端分离)

文章目录前言一、需求分析1.功能2.环境二、前端实现1.博客注册页2.博客登录页3.博客列表页4.博客详情页5.博客编辑页三、后端实现1.设计数据库表2.封装连接数据库的公共操作3.封装对博客表和用户表的操作4.实现博客列表页、博客详情页、博客编辑页的后端服务5.实现博客登录页面…

linux关于ssh免密登录、known_hosts文件

1. 关于ssh SSH 是 Secure Shell 的缩写,SSH 为建立在应用层基础上的安全协议。SSH 是目前广泛采用的安全登录协议,专为远程登录会话和其他网络服务提供安全性的协议,替代以前不安全的Telnet协议。利用 SSH 协议可以有效防止远程管理过程中的…

springboot车辆管理系统的设计与实现毕业社会源码031034

车辆管理系统的设计与实现 摘 要 科技进步的飞速发展引起人们日常生活的巨大变化,电子信息技术的飞速发展使得电子信息技术的各个领域的应用水平得到普及和应用。信息时代的到来已成为不可阻挡的时尚潮流,人类发展的历史正进入一个新时代。在现实运用中&…

java常量定义在interface和final class的区别?

问题现象: java常量定义在 interface 和 final class 的区别? 问题分析: 最近在项目开发规范中,发现有要求使用 interface类 来定义常量! 这让我想起来以前一直是用 final class 来定义常量的;那么这两者是…

【Eigen】Eigen库基础语法

这里是Eigen库的一些基础语法&#xff0c;摘自《视觉SLAM十四讲》&#xff0c;修改了书中代码的一些bug&#xff0c;部分地方添加了一些自己的理解。 头文件相关 #include <Eigen/Core> // Eigen 核心部分 #include <Eigen/Dense> // 稠密矩阵的代数运算&#xff…

Java - 通过反射进行赋值以及函数调用

Java - 通过反射进行赋值以及函数调用前言一. 通过反射进行赋值1.1 测试1.2 总结二. 通过反射进行函数调用前言 说来惭愧&#xff0c;虽然反射在Java中是非常重要和常见的一种机制。但是&#xff0c;每当自己去写这方面的代码的时候&#xff0c;总是容易愣住。还得想一想代码怎…

题库API搭建接口

题库API搭建接口 本平台优点&#xff1a;免费查题接口搭建 多题库查题、独立后台、响应速度快、全网平台可查、功能最全&#xff01; 1.想要给自己的公众号获得查题接口&#xff0c;只需要两步&#xff01; 2.题库&#xff1a;题库后台http://daili.jueguangzhe.cn/ 题库后台…

社招前端vue面试题汇总

用过pinia吗&#xff1f;有什么优点&#xff1f; 1. pinia是什么&#xff1f; 在Vue3中&#xff0c;可以使用传统的Vuex来实现状态管理&#xff0c;也可以使用最新的pinia来实现状态管理&#xff0c;我们来看看官网如何解释pinia的&#xff1a;Pinia 是 Vue 的存储库&#xff…