4

我目前正在尝试在 Go 中获取复杂结构的大小。

我已经阅读了使用reflectand的解决方案unsafe,但这些都对包含数组或映射(或指向底层数据结构的指针的任何其他字段)的结构没有帮助。

例子:

type testStruct struct {
    A     int
    B     string
    C     struct{}
    items map[string]string
}

items如果其中包含一些值,我将如何找出上面的正确字节大小?

4

1 回答 1

3

通过使用反射包,您可以非常接近结构及其内容所需的内存量。您需要遍历字段并获取每个字段的大小。例如:

func getSize(v interface{}) int {
    size := int(reflect.TypeOf(v).Size())
    switch reflect.TypeOf(v).Kind() {
    case reflect.Slice:
        s := reflect.ValueOf(v)
        for i := 0; i < s.Len(); i++ {
            size += getSize(s.Index(i).Interface())
        }
    case reflect.Map:
        s := reflect.ValueOf(v)
        keys := s.MapKeys()
        size += int(float64(len(keys)) * 10.79) // approximation from https://golang.org/src/runtime/hashmap.go
        for i := range(keys) {
            size += getSize(keys[i].Interface()) + getSize(s.MapIndex(keys[i]).Interface())
        }
    case reflect.String:
        size += reflect.ValueOf(v).Len()
    case reflect.Struct:
        s := reflect.ValueOf(v)
        for i := 0; i < s.NumField(); i++ {
            if s.Field(i).CanInterface() {
                size += getSize(s.Field(i).Interface())
            }
        }
    }
    return size
}

这使用反射获得 v 的大小,然后,对于本示例中支持的类型(切片、映射、字符串和结构),它计算存储在其中的内容所需的内存。您需要在此处添加您需要支持的其他类型。

有几个细节需要解决:

  1. 不计算私有字段。
  2. 对于结构,我们重复计算基本类型。

对于第二个,您可以在处理结构时进行递归调用之前将它们过滤掉,您可以在文档中查看反射包的种类。

于 2018-07-19T22:13:30.973 回答