一尘不染

“合并”字段将两个相同类型的结构

go

看这个struct

type Config struct {
  path string
  id   string
  key  string
  addr string
  size uint64
}

现在,我有了DefaultConfig一些值和一个从文件中加载的值的实例化FileConfig。我希望将两个结构合并到一起,以便同时获取Config两个结构的内容。FileConfig应该覆盖设置的任何内容DefaultConfig,而FileConfig
可能没有设置所有的字段 。(为什么?因为潜在的用户可能不知道默认值,所以删除该条目等同于设置默认值-我认为)

我认为我需要对此进行反思:

 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
 }

在这里我不确定:

  • 如果根本需要反思
  • 可能有更简单的方法可以做到这一点

我在这里看到的另一个问题是检查零值可能很棘手:如果覆盖结构 打算 用零值覆盖怎么办?幸运的是,我认为这不适用于我的情况-
但这变成了函数,以后可能会成为问题


阅读 237

收藏
2020-07-02

共1个答案

一尘不染

前言:
encoding/json包使用反射(包reflect)到读/写值,包括结构。其他也使用反射的库(例如TOML和YAML的实现)可能以类似的方式(或什至以相同的方式)运行,因此此处介绍的原理也适用于那些库。您需要使用所使用的库对其进行测试。

为简单起见,此处介绍的解决方案使用标准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"}`

该文件的配置覆盖S2S3S5领域。

加载配置的代码:

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 默认情况下为零,文件中缺失,结果为零
  • S2 默认为零,在文件中给定,结果是文件值
  • S3 在配置中给出,在文件中被重写为零,结果为零
  • S4 在配置中给出,在文件中丢失,结果是默认值
  • S5 在配置中给出,在文件中给出,结果是文件值
2020-07-02