Go实现一个配置包详解_第1页
Go实现一个配置包详解_第2页
Go实现一个配置包详解_第3页
Go实现一个配置包详解_第4页
Go实现一个配置包详解_第5页
已阅读5页,还剩3页未读 继续免费阅读

下载本文档

版权说明:本文档由用户提供并上传,收益归属内容提供方,若内容存在侵权,请进行举报或认领

文档简介

第Go实现一个配置包详解目录需求config包实现反序列化序列化通过环境变量/命令行参数指定配置文件封装反序列化序列化统一出口函数config包使用总结在现代软件开发中,配置文件是不可或缺的一部分。在编写Go项目时,不管是一个简单的单文件脚本还是一个庞大的微服务项目,程序的灵活性和可扩展性都需要依赖于配置文件的加载。配置文件可以包含程序的各种设置,如端口号、数据库连接信息、认证密钥等,这些设置可能需要根据不同的环境和部署情况进行调整。因此,一个可靠的配置文件加载机制对于项目的成功实现至关重要。本文就来探究下在Go项目中如何更加方便的加载和管理配置。

在Go中,有很多方法可以加载和管理配置,例如使用命令行标志、环境变量或者读取特定格式的文件。选择哪种方法取决于项目的具体需求和开发人员的偏好。

需求

根据我的开发经验,一个配置包通常需要支持如下功能:

支持从环境变量、命令行标志加载配置文件,然后将配置文件内容映射到Go结构体中,这个过程叫作反序列化。

支持将Go结构体中的配置信息写入配置文件,这个过程叫作序列化。

起码要支持YAML、JSON这两种最常见的配置文件格式。

config包实现

反序列化

反序列化操作是配置包最常用的功能,通过反序列化我们可以将配置文件内容映射到Go结构体中。所以我们先来实现如何进行配置文件的反序列化操作。

typeFileTypeint

const(

FileTypeYAML=iota

FileTypeJSON

funcLoadConfig(filenamestring,cfginterface{},typFileType)error{

data,err:=os.ReadFile(filename)

iferr!=nil{

returnfmt.Errorf("ReadFile:%v",err)

switchtyp{

caseFileTypeYAML:

returnyaml.Unmarshal(data,cfg)

caseFileTypeJSON:

returnjson.Unmarshal(data,cfg)

returnerrors.New("unsupportedfiletype")

}

首先定义一个FileType类型,用来区分不同的配置文件格式,我们要实现的日志包能够支持YAML和JSON两种格式。

接下来实现了LoadConfig函数,用来加载配置,它接收三个参数:配置文件名称、配置结构体、文件类型。

在函数内部,首先使用os.ReadFile(filename)读取配置文件内容,然后根据文件类型,使用不同的方法来反序列化,最终将配置信息映射到cfg结构体中。

序列化

配置包还要支持序列化操作,也就是将配置信息从内存中写入文件。有时候我们想保存一个example-config.yaml供项目使用者参考,这个功能就很实用。

funcDumpConfig(filenamestring,cfginterface{},typFileType)error{

var(

data[]byte

errerror

switchtyp{

caseFileTypeYAML:

data,err=yaml.Marshal(cfg)

caseFileTypeJSON:

data,err=json.Marshal(cfg)

iferr!=nil{

returnerr

f,err:=os.Create(filename)

iferr!=nil{

returnerr

_,err=f.Write(data)

_=f.Close()

returnerr

}

我们实现了DumpConfig函数用来进行序列化操作,该函数同样接收三个参数:配置文件名、配置结构体、文件类型。

在函数中,首先根据文件类型对配置对象cfg进行序列化操作得到data,然后通过os.Create(filename)创建配置文件文件,最后使用f.Write(data)将序列化后的内容写入文件。

现在配置包的核心功能已经实现了,并且两个函数都被允许导出,如果其他项目想使用,其实是可以拿过去直接使用的。不过目前的配置包在易用性上还有可优化的空间。

通过环境变量/命令行参数指定配置文件

以上序列化/反序列化函数的实现,配置文件名都是需要通过参数传递进来的,我们还可以通过环境变量或者命令行参数的形式来指定配置文件名。

var(

cfgPath=flag.String("c","config.yaml","pathtoconfigfile")

dump=flag.Bool("d",false,"dumpconfigtofile")

funcinit(){

_=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配置,这样调用方在使用时能够少传递一个参数。实现如下:

funcLoadYAMLConfig(filenamestring,cfginterface{})error{

returnLoadConfig(filename,cfg,FileTypeYAML)

funcLoadJSONConfig(filenamestring,cfginterface{})error{

returnLoadConfig(filename,cfg,FileTypeJSON)

}

其次,我们拿到的cfgPath参数也能派上用场了,可以对LoadYAMLConfig、LoadJSONConfig进一步封装:

funcLoadYAMLConfigFromFlag(cfginterface{})error{

if!flag.Parsed(){

flag.Parse()

returnLoadYAMLConfig(*cfgPath,cfg)

funcLoadJSONConfigFromFlag(cfginterface{})error{

if!flag.Parsed(){

flag.Parse()

returnLoadJSONConfig(*cfgPath,cfg)

}

因为cfgPath是通过环境变量或命令行参数形式传入,所以调用反序列化函数的时候也就没必要手动传入了。

不过在使用cfgPath变量之前,我们调用了flag.Parsed()并对其返回值进行了判断,如果flag.Parsed()返回false则调用flag.Parse()。

这一步操作是因为,配置包的调用方可能也使用了flag包,那么如果调用方已经调用过flag.Parse()解析了命令行参数,调用flag.Parsed()就会返回true,此时配置包内部没必要再解析一遍命令行参数,可以直接使用。

序列化

封装好了反序列化操作,对应的,也要封装序列化操作。代码实现上与反序列化操作如出一辙,我将代码贴在这里,就不再进行讲解了。

funcDumpYAMLConfig(filenamestring,cfginterface{})error{

returnDumpConfig(filename,cfg,FileTypeYAML)

funcDumpJSONConfig(filenamestring,cfginterface{})error{

returnDumpConfig(filename,cfg,FileTypeJSON)

funcDumpYAMLConfigFromFlag(cfginterface{})error{

if!flag.Parsed(){

flag.Parse()

returnDumpYAMLConfig(*cfgPath,cfg)

funcDumpJSONConfigFromFlag(cfginterface{})error{

if!flag.Parsed(){

flag.Parse()

returnDumpJSONConfig(*cfgPath,cfg)

}

统一出口函数

我们通过环境变量或命令行参数拿到的另外一个用户指定的变量dump还没有使用,dump是一个bool值,代表用户是否想要序列化配置,如果为true则进行序列化操作,否则进行反序列化操作。

由此,我们可以将序列化/反序列化操作封装成一个函数,内部通过dump的值来决定执行哪种操作。实现如下:

funcLoadOrDumpYAMLConfigFromFlag(cfginterface{})error{

if!flag.Parsed(){

flag.Parse()

if*dump{

iferr:=DumpYAMLConfig(*cfgPath,cfg);err!=nil{

returnerr

os.Exit(0)

returnLoadYAMLConfig(*cfgPath,cfg)

funcLoadOrDumpJSONConfigFromFlag(cfginterface{})error{

if*dump{

iferr:=DumpJSONConfigFromFlag(cfg);err!=nil{

returnerr

os.Exit(0)

returnLoadJSONConfigFromFlag(cfg)

}

我们对YAML和JSON格式的配置分别实现了LoadOrDumpYAMLConfigFromFlag和LoadOrDumpJSONConfigFromFlag函数。顾名思义,这个函数既能加载(Load)配置,也能将配置写入(Dump)文件。

两个函数内部思路相同,如果dump为true,就序列化配置信息到文件中,并且使用os.Exit(0)退出程序。此时,控制台不会有任何输出,这遵循了Linux/Unix系统的设计哲学,没有消息就是最好的消息;如果dump为false,则进行反序列化操作。

以上我们就实现了自定义配置包的全部功能,完整代码可以点击这里查看。

config包使用

配置包的完整使用示例我写在了项目GitHub仓库的RE

温馨提示

  • 1. 本站所有资源如无特殊说明,都需要本地电脑安装OFFICE2007和PDF阅读器。图纸软件为CAD,CAXA,PROE,UG,SolidWorks等.压缩文件请下载最新的WinRAR软件解压。
  • 2. 本站的文档不包含任何第三方提供的附件图纸等,如果需要附件,请联系上传者。文件的所有权益归上传用户所有。
  • 3. 本站RAR压缩包中若带图纸,网页内容里面会有图纸预览,若没有图纸预览就没有图纸。
  • 4. 未经权益所有人同意不得将文件中的内容挪作商业或盈利用途。
  • 5. 人人文库网仅提供信息存储空间,仅对用户上传内容的表现方式做保护处理,对用户上传分享的文档内容本身不做任何修改或编辑,并不能对任何下载内容负责。
  • 6. 下载文件中如有侵权或不适当内容,请与我们联系,我们立即纠正。
  • 7. 本站不保证下载资源的准确性、安全性和完整性, 同时也不承担用户因使用这些下载资源对自己和他人造成任何形式的伤害或损失。

评论

0/150

提交评论