Golang 使用面向对象思想编程 2

目的是写一个配置文件保存模块,保存工程中某些需要保存的配置。
由于需要保存的数据比较零散,分散在很多模块中。有想过建一个大结构体,然后把所有需要保存的数据放到这个结构体中,但是在保存和读取时,需要把其它模块的数据转到这个大结构体(模块多了,操作起来很不方便)。另外一种想法就是,直接把其它模块需要保存的数据定义在这个模块中,这样会破坏模块的完整性。

然后,我选择了第一种方法。写完发现package 循环导入的问题。包B是我的配置保存模块,里面有包A的数据结构,所以包B需要导入包A。但是包A需要使用包B的保存方法,包A就需要导入包B。然后就GG了。。。

这个时候如果考虑第二种方法,把数据结构定义到配置保存模块中,那么,就不会发生包循环导入的问题了。但是,我之前写的模块就被破坏了,我要把定义的结构体移到保存配置模块中。然后,程序员的执着不允许我这么做。

然后,使用面向对象的思想,我在保存配置的模块定义了一个结构,包含读配置和写配置。还定义了一个结构,用来保存所有的配置数据,来自不同模块的,为了不循环导入,因此使用空接口。这里,我之定义了两个空接口,因为只有两个包的配置数据需要保存。

type GetConfiger interface {
	GetConfig(*GlobalConfig) error
	SetConfig(*GlobalConfig, GetConfiger)
}

/* 使用空接口,解决package循环导入的问题 */
type GlobalConfig struct {
	LogConfig interface{} `json:"LogConfig"`
	ApnParam  interface{} `json:"ApnParam"`
}

结果一这样定义,我就感觉这个结构体可以储存任何配置了。
最后一通折腾,代码如下:

package file

import (
	"encoding/json"
	"errors"
	"os"

	log "github.com/sirupsen/logrus"
)

type GetConfiger interface {
	GetConfig(*GlobalConfig) error
	SetConfig(*GlobalConfig, GetConfiger)
}

/* 使用空接口,解决package循环导入的问题 */
type GlobalConfig struct {
	LogConfig interface{} `json:"LogConfig"`
	ApnParam  interface{} `json:"ApnParam"`
}

const (
	GLOBALCONFIG_FILE = "global.conf"
)

/* 全局配置 */
var gblConf = &GlobalConfig{}

func init() {
	ReadFile()
}

func Write(conf GetConfiger) {
	conf.SetConfig(gblConf, conf)
	data, err := json.MarshalIndent(gblConf, "", "\t")
	if err != nil {
		log.Error(err)
	}

	err = os.WriteFile(GLOBALCONFIG_FILE, data, 0666)
	if err != nil {
		log.Error(err)
	}
}

func Read(g GetConfiger) error {
	return g.GetConfig(gblConf)
}

func ReadFile() {
	data, err := os.ReadFile(GLOBALCONFIG_FILE)
	if err != nil {
		log.Warn(err)
		return
	}

	err = json.Unmarshal(data, gblConf)
	if err != nil {
		log.Warn(err)
		return
	}
}

func (gb *GlobalConfig) GetConfigToStruct(gi GetConfiger, gs interface{}) error {
	if gs == nil {
		err := errors.New("interface{} is nil for parse")
		log.Warn(err)
		return err
	}
	data, err := json.Marshal(gs)
	if err != nil {
		log.Error(err)
		return err
	}

	err = json.Unmarshal(data, gi)
	if err != nil {
		log.Error(err)
	}
	return err
}

然后我的日志模块,定义的数据类型如下。

type LogConfig struct {
	/* 日志文件句柄 */
	logFile *os.File `json:"-"`
	/* 日志格式采用 “text” 或 “json” */
	Format string `json:"Format"`
	/* 日志输出到文件 */
	ToFile bool `json:"ToFile"`
	/* 日志输出到终端 */
	ToStdout bool `json:"ToStdout"`
	/* 日志等级 */
	Level string `json:"Level"`
	/* 日志文件名称 */ 
	OutFile string `json:"OutFile"`
	/* 日志文件的大小限制 unit: Byte */
	MaxSize uint32 `json:"MaxSize"`
	/* 日志格式控制 */
	Formatter *MyFormatter `json:"Formatter"`
}

type MyFormatter struct {
	/* 打印调用信息,文件 行 */
	ReportCaller bool
	/* 显示等级信息 */
	ShowLevel bool
	/* 日志带颜色 */
	ColorEnable bool
	/* 显示时间戳 */
	Timestamp bool
	/* 时间戳格式 */
	TimestampFormat  string
	callerPrettyfier func(*runtime.Frame) (file string, function string, line string) `json:"-"`
}

这个时候要使用配置保存模块,只需要下面几行代码就可以了。

  • 实现了GetConfiger接口的两个方法。这里的接口转结构体代码已经在我的模块中写好了通用的。直接调用就行了。SetConfig方法需要实现,将结构体保存到GlobalConfig结构体中的哪一个字段就行了,这决定了这个配置将使用哪一个字段来作为json的键。
func (lc *LogConfig) GetConfig(g *file.GlobalConfig) error {
	return g.GetConfigToStruct(lc, g.LogConfig)
}

func (lc *LogConfig) SetConfig(g *file.GlobalConfig, conf file.GetConfiger) {
	g.LogConfig = *lc
}
  • 使用
func main() {
	lc := &LogConfig{}
	file.Read(lc)
	lc.ToStdout = false
	file.Write(lc)
}

需要添加配置在GlobalConfig结构体中添加字段,并实现GetConfiger接口的两个方法(只需要照着上面的实现复制粘贴就行了,就2行代码),就可以愉快的使用WriteRead方法来保存和读取配置了。

两外一个模块使用保存配置模块的案例。

type APN_param struct {
	Apn     string
	User    string
	Passwd  string
	Dialnum string
}

func (a *APN_param) GetConfig(g *file.GlobalConfig) error {
	return g.GetConfigToStruct(a, g.ApnParam)
}

func (a *APN_param) SetConfig(g *file.GlobalConfig, conf file.GetConfiger) {
	g.ApnParam = *a
}

func main() {
	apn := APN_param{}
	file.Read(&apn)
	apn.Apn = "CMCC"
	file.Write(&apn)
}

完美解决所有问题。既不会循环导入,数据和操作也是模块分离的,还支持在不修改代码的情况下实现所有自定义类型的配置保存,使用起来也非常的方便。

-----新增-----
保存配置模块在需要添加许多结构体数据保存时,仍然要修改一下结构体GlobalConfig的内容,因此考虑使用空接口数组或者空接口映射来实现添加任意多的内容。

通过使用map[string]interface{}类型,实现接口方法时,指定自己配置保存使用的键即可。以下是究极完美版,使用这个模块时,再不用对这个模块进行任何代码修改。
代码如下:

package file

import (
	"encoding/json"
	"errors"
	"os"

	log "github.com/sirupsen/logrus"
)

type GetConfiger interface {
	GetConfig(*GlobalConfig) error
	SetConfig(*GlobalConfig, GetConfiger)
}

type GlobalConfig map[string]interface{}

const (
	GLOBALCONFIG_FILE = "global.json"
)

/* 全局配置 */
var gblConf = &GlobalConfig{}

func init() {
	ReadFile()
}

func Write(conf GetConfiger) {
	conf.SetConfig(gblConf, conf)
	data, err := json.MarshalIndent(gblConf, "", "\t")
	if err != nil {
		log.Error(err)
	}
	err = os.WriteFile(GLOBALCONFIG_FILE, data, 0666)
	if err != nil {
		log.Error(err)
	}
}

func Read(g GetConfiger) error {
	return g.GetConfig(gblConf)
}

func ReadFile() {
	data, err := os.ReadFile(GLOBALCONFIG_FILE)
	if err != nil {
		log.Warn(err)
		return
	}
	err = json.Unmarshal(data, gblConf)
	if err != nil {
		log.Warn(err)
		return
	}
}

func (gb *GlobalConfig) GetConfigToStruct(gi GetConfiger, key string) error {
	gs := map[string]interface{}(*gb)[key]
	if gs == nil {
		err := errors.New("interface{} is nil for parse")
		log.Warn(err)
		return err
	}
	data, err := json.Marshal(gs)
	if err != nil {
		log.Error(err)
		return err
	}
	err = json.Unmarshal(data, gi)
	if err != nil {
		log.Error(err)
	}
	return err
}

func (gb *GlobalConfig) Set(key string, gi GetConfiger) {
	map[string]interface{}(*gb)[key] = gi
}

使用时,直接调用GlobalConfig的两个方法即可,参数为当前结构保存时使用的键:

func (lc *LogConfig) GetConfig(g *file.GlobalConfig) error {
	return g.GetConfigToStruct(lc, "LogConfig")
}

func (lc *LogConfig) SetConfig(g *file.GlobalConfig, conf file.GetConfiger) {
	g.Set("LogConfig", lc)
}

-------------再来一个牛逼的版本------------------
去掉GetConfigSetConfig方法,直接使用空接口。这样的好处就是,其它需要存储的数据,可以不用实现这两个方法,也能直接用WriteRead。但是需要存储的结构体就没有办法和存储后的数据进行关联了,因此,需要在WriteRead方法参数列表加上需要关联的键,用于对数据的正确处理。
代码如下:

package file

import (
	"encoding/json"
	"errors"
	"os"

	log "github.com/sirupsen/logrus"
)

type GetConfiger interface {
}

type GlobalConfig map[string]interface{}

const (
	GLOBALCONFIG_FILE = "global.json"
)

/* 全局配置 */
var gblConf = &GlobalConfig{}

func init() {
	ReadFile()
}

func Write(conf GetConfiger, key string) {
	gblConf.Set(key, conf)
	data, err := json.MarshalIndent(gblConf, "", "\t")
	if err != nil {
		log.Error(err)
		return
	}
	err = os.WriteFile(GLOBALCONFIG_FILE, data, 0666)
	if err != nil {
		log.Error(err)
	}
}

func Read(g GetConfiger, key string) error {
	return gblConf.GetConfigToStruct(g, key)
}

func ReadFile() {
	data, err := os.ReadFile(GLOBALCONFIG_FILE)
	if err != nil {
		log.Warn(err)
		return
	}
	err = json.Unmarshal(data, gblConf)
	if err != nil {
		log.Warn(err)
		return
	}
}

func (gb *GlobalConfig) GetConfigToStruct(gi GetConfiger, key string) error {
	gs := map[string]interface{}(*gb)[key]
	if gs == nil {
		err := errors.New("interface{} is nil for parse")
		log.Warn(err)
		return err
	}
	data, err := json.Marshal(gs)
	if err != nil {
		log.Error(err)
		return err
	}
	err = json.Unmarshal(data, gi)
	if err != nil {
		log.Error(err)
	}
	return err
}

func (gb *GlobalConfig) Set(key string, gi GetConfiger) {
	map[string]interface{}(*gb)[key] = gi
}

使用:
这种方法处理的好处就是,再也不用写任何代码了,直接用就行了。如下:

func main() {
	file.ReadFile()
	apn := APN_param{}
	file.Read(&apn, "APN")
	apn.Apn = "CMCC"
	file.Write(&apn, "APN")
	
	lc := &LogConfig{}
	file.Read(lc, "LogConfig")
	lc.ToStdout = false
	file.Write(lc, "LogConfig")
}

在这里插入图片描述

最后面的方式,只在这个结构体保存json的场景下使用,因为,标准库的json包可以自动给我进行转化,我们不用关心数据的具体情况。前面的方式是比较通用的,可以处理问题比较多。

寥寥几十行代码,就可以做到保存和读取任意配置。不得不说Golang真的强。

posted @ 2021-11-16 23:55  duapple  阅读(4)  评论(0编辑  收藏  举报  来源