看看这struct个:

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

现在我有一个用一些值初始化的DefaultConfig和一个从文件加载的值,比方说FileConfig. 我希望将这两个 struct 合并,这样我就可以得到包含两个 struct 内容的Config.FileConfig应覆盖在DefaultConfig中设置的任何内容,而FileConfig may not have all fields set. (为什么是这个?因为潜在用户可能不知道默认值,所以删除该条目相当于设置默认值-我认为)

我想我需要反省一下:

 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
 }

在这里我不太确定:

  • 如果完全需要反射
  • 也许有更简单的方法可以做到这一点.

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

推荐答案

Foreword: encoding/json包使用反射(包reflect)读/写值,包括 struct .其他也使用反射的库(如TOML和YAML的实现)可能会以类似(甚至相同的方式)操作,因此这里介绍的原则也可能适用于这些库.您需要使用您使用的库对其进行测试.

为简单起见,这里提供的解决方案使用标准库的encoding/json.


一个优雅的、"零努力"的解决方案是使用encoding/jsonunmarshal into a value of the "prepared", default configuration.

这可以处理你需要的一切:

  • 配置文件中缺少值:默认值适用
  • 文件中给出的值会覆盖默认配置(无论它是什么)
  • 文件中零值的显式覆盖优先(覆盖非零默认配置)

为了进行演示,我们将使用以下配置 struct :

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上try ):

&{S1: S2:file-s2 S3: S4:def S5:file-s5}

分析结果:

  • S1默认为零,文件中缺少,结果为零
  • S2默认为零,在文件中给出,结果为文件值
  • S3在配置中给定,在文件中被重写为零,结果为零
  • 配置中给出了S4,文件中缺少,结果是默认值
  • S5在配置中给出,在文件中给出,结果是文件值

Go相关问答推荐

从Kafka到Clickhouse的实时消费数据

读取JSON数据并在网页上显示

为什么工具链指令在这种情况下没有效果?

为什么我不能使用Docker从本地访问我的Gin应用程序?

更改位置级别和时间戳零点Golang

在Mac中使用uname获取处理器体系 struct 时,在为AMD64构建Go二进制时出现错误结果

创建使用逗号而不是加号分隔OU的CSR

为什么 `go mod` 占用了另一个磁盘上的空间而不是我的 GOPATH?

当我有外键时,如何使用 GORM 创建数据库的新条目

最长连续重复的字符golang

使用 Go Colly 抓取所有可能的标签并将它们放入一个变量中

有没有办法约束(通用)类型参数?

将 big.Int 转换为 [2]int64,反之亦然和二进制补码

如何使用 go-playground/validator 编写 snake case 绑定标签?

使用 go.work 文件在多个测试文件上运行 go test 命令

如何根据 Go 中第二次出现的分隔符拆分字符串?

为什么我不能使用来自 gocloak 中 Login() 的访问令牌在 KeyCloak 中创建新客户端?

GORM GIN 更新查询导致 500 内部服务器错误

带有 grpc 的 protobuf 用于拆分包中的 Go

如何使用通用字段初始化匿名struct数组