看这个struct:
struct
type Config struct { path string id string key string addr string size uint64 }
现在,我有了DefaultConfig一些值和一个从文件中加载的值的实例化FileConfig。我希望将两个结构合并到一起,以便同时获取Config两个结构的内容。FileConfig应该覆盖设置的任何内容DefaultConfig,而FileConfig 可能没有设置所有的字段 。(为什么?因为潜在的用户可能不知道默认值,所以删除该条目等同于设置默认值-我认为)
DefaultConfig
FileConfig
Config
我认为我需要对此进行反思:
func merge(default *Config, file *Config) (*Config) { b := reflect.ValueOf(default).Elem() o := reflect.ValueOf(file).Elem() for i := 0; i < b.NumField(); i++ { defaultField := b.Field(i) fileField := o.Field(i) if defaultField.Interface() != reflect.Zero(fileField.Type()).Interface() { defaultField.Set(reflect.ValueOf(fileField.Interface())) } } return default }
在这里我不确定:
我在这里看到的另一个问题是检查零值可能很棘手:如果覆盖结构 打算 用零值覆盖怎么办?幸运的是,我认为这不适用于我的情况- 但这变成了函数,以后可能会成为问题
前言: 该encoding/json包使用反射(包reflect)到读/写值,包括结构。其他也使用反射的库(例如TOML和YAML的实现)可能以类似的方式(或什至以相同的方式)运行,因此此处介绍的原理也适用于那些库。您需要使用所使用的库对其进行测试。
encoding/json
reflect
为简单起见,此处介绍的解决方案使用标准lib的encoding/json。
一个优雅且“零努力”的解决方案是使用该encoding/json程序包并将其 解编为“ prepared”(默认)配置的值 。
这可以处理您需要的一切:
为了演示,我们将使用以下配置结构:
type Config struct { S1 string S2 string S3 string S4 string S5 string }
和默认配置:
var defConfig = &Config{ S1: "", // Zero value S2: "", // Zero value S3: "abc", S4: "def", S5: "ghi", }
假设该文件包含以下配置:
const fileContent = `{"S2":"file-s2","S3":"","S5":"file-s5"}`
该文件的配置覆盖S2,S3和S5领域。
S2
S3
S5
加载配置的代码:
conf := new(Config) // New config *conf = *defConfig // Initialize with defaults err := json.NewDecoder(strings.NewReader(fileContent)).Decode(&conf) if err != nil { panic(err) } fmt.Printf("%+v", conf)
和输出(在Go Playground上尝试):
&{S1: S2:file-s2 S3: S4:def S5:file-s5}
分析结果:
S1
S4