Go:使用不同数字类型的切片作为结构的字段

Go: use slice of different numeric types as field of a struct

我正在编写一个收集器,用于收集指标并将其存储在如下所示的结构中:

type Metric struct {
  Name string
  Data []float64
}

但是对于某些指标,使用 float64 没有意义,因为它们的值是无符号整数。知道如何为 Data 字段使用不同的数字类型吗?

我可以使用 Data []interface{},但我将无法对数组元素使用索引。

(为清楚起见:我不需要一个切片中的不同类型,例如 Python 中的列表:我的切片必须是强类型的,但我希望能够更改切片。)

要获得完整的解决方案,您必须等到泛型登陆 Go(可能在 1.18 中):https://blog.golang.org/generics-proposal

使用泛型,您将能够拥有一个泛型 Metric 类型,它可以包含 float64unsigned,并且您可以分别实例化它们中的每一个。

例如(generics-enabled playgorund):

type Metric[T any] struct {
  Name string
  Data []T
}

func main() {
  mf := Metric[float64]{"foo", []float64{12.24, 1.1, 2.22}}
  mu := Metric[uint32]{"bar", []uint32{42, 2}}

  fmt.Println(mf)
  fmt.Println(mu)
}

注意 [T any] 表示 Data 中保存的类型不受约束。您可以将其限制为具有某些特征的类型,或者如果您愿意,可以将其限制为像 float64, uint32 这样的硬编码列表。


同时,还有一些选择:

  1. float64可以表示很多整数;至少所有 32 位的(参见 Representing integers in doubles
  2. 可以使用Data []interface{},但它相当浪费。对该切片进行索引应该没有问题,但是无论何时使用它都必须有类型断言。它在内存方面和运行时性能方面都很昂贵;对指标真正重要的东西。
  3. 您可以拥有两个版本的 Metric,代码重复(并在需要时使用代码生成来提供帮助)。