Go实现一个配置包详解
作者:江湖十年 发布时间:2024-05-22 10:29:57
在现代软件开发中,配置文件是不可或缺的一部分。在编写 Go 项目时,不管是一个简单的单文件脚本还是一个庞大的微服务项目,程序的灵活性和可扩展性都需要依赖于配置文件的加载。配置文件可以包含程序的各种设置,如端口号、数据库连接信息、认证密钥等,这些设置可能需要根据不同的环境和部署情况进行调整。因此,一个可靠的配置文件加载机制对于项目的成功实现至关重要。本文就来探究下在 Go 项目中如何更加方便的加载和管理配置。
在 Go 中,有很多方法可以加载和管理配置,例如使用命令行标志、环境变量或者读取特定格式的文件。选择哪种方法取决于项目的具体需求和开发人员的偏好。
需求
根据我的开发经验,一个配置包通常需要支持如下功能:
支持从环境变量、命令行标志加载配置文件,然后将配置文件内容映射到 Go 结构体中,这个过程叫作反序列化。
支持将 Go 结构体中的配置信息写入配置文件,这个过程叫作序列化。
起码要支持 YAML、JSON 这两种最常见的配置文件格式。
config 包实现
反序列化
反序列化操作是配置包最常用的功能,通过反序列化我们可以将配置文件内容映射到 Go 结构体中。所以我们先来实现如何进行配置文件的反序列化操作。
type FileType int
const (
FileTypeYAML = iota
FileTypeJSON
)
func LoadConfig(filename string, cfg interface{}, typ FileType) error {
data, err := os.ReadFile(filename)
if err != nil {
return fmt.Errorf("ReadFile: %v", err)
}
switch typ {
case FileTypeYAML:
return yaml.Unmarshal(data, cfg)
case FileTypeJSON:
return json.Unmarshal(data, cfg)
}
return errors.New("unsupported file type")
}
首先定义一个 FileType
类型,用来区分不同的配置文件格式,我们要实现的日志包能够支持 YAML 和 JSON 两种格式。
接下来实现了 LoadConfig
函数,用来加载配置,它接收三个参数:配置文件名称、配置结构体、文件类型。
在函数内部,首先使用 os.ReadFile(filename)
读取配置文件内容,然后根据文件类型,使用不同的方法来反序列化,最终将配置信息映射到 cfg
结构体中。
序列化
配置包还要支持序列化操作,也就是将配置信息从内存中写入文件。有时候我们想保存一个 example-config.yaml
供项目使用者参考,这个功能就很实用。
func DumpConfig(filename string, cfg interface{}, typ FileType) error {
var (
data []byte
err error
)
switch typ {
case FileTypeYAML:
data, err = yaml.Marshal(cfg)
case FileTypeJSON:
data, err = json.Marshal(cfg)
}
if err != nil {
return err
}
f, err := os.Create(filename)
if err != nil {
return err
}
_, err = f.Write(data)
_ = f.Close()
return err
}
我们实现了 DumpConfig
函数用来进行序列化操作,该函数同样接收三个参数:配置文件名、配置结构体、文件类型。
在函数中,首先根据文件类型对配置对象 cfg
进行序列化操作得到 data
,然后通过 os.Create(filename)
创建配置文件文件,最后使用 f.Write(data)
将序列化后的内容写入文件。
现在配置包的核心功能已经实现了,并且两个函数都被允许导出,如果其他项目想使用,其实是可以拿过去直接使用的。不过目前的配置包在易用性上还有可优化的空间。
通过环境变量/命令行参数指定配置文件
以上序列化/反序列化函数的实现,配置文件名都是需要通过参数传递进来的,我们还可以通过环境变量或者命令行参数的形式来指定配置文件名。
var (
cfgPath = flag.String("c", "config.yaml", "path to config file")
dump = flag.Bool("d", false, "dump config to file")
)
func init() {
_ = flag.Set("c", os.Getenv("CONFIG_PATH"))
_ = flag.Set("d", os.Getenv("DUMP_CONFIG"))
}
命令行参数解析可以使用 Go 内置的 flag 包来实现,我们在这里声明了两个标志,-c
用来传递配置文件名,-d
用来决定要进行序列化还是反序列化操作。
此外,在 init
函数中,我们实现了从环境变量中加载配置文件名和需要执行的操作。当配置包被导入时,会优先执行 init
函数,所以这些信息会优先从环境变量读取。
封装
现在我们可以通过环境变量或命令行参数的形式拿到配置文件名和需要执行的操作两个参数,并且分别赋值给 cfgPath
、dump
两个变量,接下来我们将对之前实现的序列化/反序列化函数进行封装,让其使用起来更加便捷。
反序列化
首先,我们可以将反序列化操作函数 LoadConfig
进一步封装,将其包装成两个函数分别加载 YAML 配置和 JSON 配置,这样调用方在使用时能够少传递一个参数。实现如下:
func LoadYAMLConfig(filename string, cfg interface{}) error {
return LoadConfig(filename, cfg, FileTypeYAML)
}
func LoadJSONConfig(filename string, cfg interface{}) error {
return LoadConfig(filename, cfg, FileTypeJSON)
}
其次,我们拿到的 cfgPath
参数也能派上用场了,可以对 LoadYAMLConfig
、LoadJSONConfig
进一步封装:
func LoadYAMLConfigFromFlag(cfg interface{}) error {
if !flag.Parsed() {
flag.Parse()
}
return LoadYAMLConfig(*cfgPath, cfg)
}
func LoadJSONConfigFromFlag(cfg interface{}) error {
if !flag.Parsed() {
flag.Parse()
}
return LoadJSONConfig(*cfgPath, cfg)
}
因为 cfgPath
是通过环境变量或命令行参数形式传入,所以调用反序列化函数的时候也就没必要手动传入了。
不过在使用 cfgPath
变量之前,我们调用了 flag.Parsed()
并对其返回值进行了判断,如果 flag.Parsed()
返回 false
则调用 flag.Parse()
。
这一步操作是因为,配置包的调用方可能也使用了 flag 包,那么如果调用方已经调用过 flag.Parse()
解析了命令行参数,调用 flag.Parsed()
就会返回 true
,此时配置包内部没必要再解析一遍命令行参数,可以直接使用。
序列化
封装好了反序列化操作,对应的,也要封装序列化操作。代码实现上与反序列化操作如出一辙,我将代码贴在这里,就不再进行讲解了。
func DumpYAMLConfig(filename string, cfg interface{}) error {
return DumpConfig(filename, cfg, FileTypeYAML)
}
func DumpJSONConfig(filename string, cfg interface{}) error {
return DumpConfig(filename, cfg, FileTypeJSON)
}
func DumpYAMLConfigFromFlag(cfg interface{}) error {
if !flag.Parsed() {
flag.Parse()
}
return DumpYAMLConfig(*cfgPath, cfg)
}
func DumpJSONConfigFromFlag(cfg interface{}) error {
if !flag.Parsed() {
flag.Parse()
}
return DumpJSONConfig(*cfgPath, cfg)
}
统一出口函数
我们通过环境变量或命令行参数拿到的另外一个用户指定的变量 dump
还没有使用,dump
是一个 bool
值,代表用户是否想要序列化配置,如果为 true
则进行序列化操作,否则进行反序列化操作。
由此,我们可以将序列化/反序列化操作封装成一个函数,内部通过 dump
的值来决定执行哪种操作。实现如下:
func LoadOrDumpYAMLConfigFromFlag(cfg interface{}) error {
if !flag.Parsed() {
flag.Parse()
}
if *dump {
if err := DumpYAMLConfig(*cfgPath, cfg); err != nil {
return err
}
os.Exit(0)
}
return LoadYAMLConfig(*cfgPath, cfg)
}
func LoadOrDumpJSONConfigFromFlag(cfg interface{}) error {
if *dump {
if err := DumpJSONConfigFromFlag(cfg); err != nil {
return err
}
os.Exit(0)
}
return LoadJSONConfigFromFlag(cfg)
}
我们对 YAML 和 JSON 格式的配置分别实现了 LoadOrDumpYAMLConfigFromFlag
和 LoadOrDumpJSONConfigFromFlag
函数。顾名思义,这个函数既能加载(Load
)配置,也能将配置写入(Dump
)文件。
两个函数内部思路相同,如果 dump
为 true
,就序列化配置信息到文件中,并且使用 os.Exit(0)
退出程序。此时,控制台不会有任何输出,这遵循了 Linux/Unix 系统的设计哲学,没有消息就是最好的消息;如果 dump
为 false
,则进行反序列化操作。
以上我们就实现了自定义配置包的全部功能,完整代码可以点击这里查看。
config 包使用
配置包的完整使用示例我写在了项目 GitHub 仓库的 README.md 文件中,你可以点进去查看。
来源:https://juejin.cn/post/7226385225579348023


猜你喜欢
- 1、复杂SQL查询1.1、单表查询(1)选择指定的列[例]查询全体学生的学号和姓名select Sno as 学号,Sname as 姓名
- 用了on error resume next则在这句往后的代码就算出错也会继续执行具体有没有错可以用err.number来判断err.num
- SQL Server数据库查询速度慢的原因有很多,常见的有以下几种:1、没有索引或者没有用到索引(这是查询慢最常见的问题,是程序设计的缺陷)
- 当前需求:有group和factor两张表,一个group对应多个factor,现在想查询有效的group和对应的有效的factor,两个表
- jsp表达式方式: <center> <table border="1"> <% for
- 本文实例讲述了Python实现手写一个类似django的web框架。分享给大家供大家参考,具体如下:用与django相似结构写一个web框架
- 当有两个表,例如一个学生表,一个班级表,是多对一的关系。方法1:c = models.Class.object.get(pk=1)#查询到I
- Win10系统安装MySQL8.0遇到的问题及解决方法,具体内容如下所示:对着第一个桌面应用击右键,选择“以管理员身份运行”选项,就可以以管
- 本文为大家分享了pygame游戏之旅的第6篇,供大家参考,具体内容如下定义一个障碍模型函数:def things(thingx, thing
- 方案概要: 1. 改变文件存储时的文件名 2. 配置索引服务器,并将索引服务器与MS SQL Server关联。 3. 修改SQL语句,将进
- “输入框( Input )应当符合逻辑地划分为小组,这样大脑就可以很好的处理大堆区域间的关系。”– 《HTML权威指南》Web 应用程序总是
- vue单向数据流在vue中需要遵循单向数据流原则在父传子的前提下,父组件的数据发生会通知子组件自动更新子组件内部,不能直接修改父组件传递过来
- 出图是项目里常见的任务,有的项目甚至会要上百张图片,所以批量出土工具很有必要。arcpy.mapping就是ArcGIS里的出图模块,能快速
- 上节我们提到解决赋值中等号两边参数不一致的方法可以通过切片,但在Python3中我们可以利用特定的语法更加方便的处理这种情况,如下示例。当带
- 我就废话不多说了,直接上代码吧!import numpy as npimport matplotlib.pyplot as pltdef g
- 本文实例讲述了Yii框架引用插件和ckeditor中body与P标签去除的方法。分享给大家供大家参考,具体如下:在Yii中引用插件注:插件和
- 写在前面最近正好有音视频编辑的需求,虽然之前粗略的了解过FFmpeg不过肯定是不够用的,借此重新学习下;基本概念ffmpeg概念Fmpeg的
- 09年的电影缓缓的落下帷幕,以及新年伊始,轰轰烈烈催人癫狂的《阿凡达》。整年里,最让人我记忆深刻的还是《飞屋历险记》。Carl与Ellie被
- 注释:在大多数的情况下,修改MySQL是需要有mysql里的root权限的,所以一般用户无法更改密码,除非请求管理员。方法1使用phpmya
- 比如:import linecacheprint linecache.getline('2.1_open.py&