繁体   English   中英

如何通过 Go 中的反射更新结构体的数组字段值?

[英]How I can update struct's array field value via reflect in Go?

我想将 struct 字段的值编辑为 interface{} 参数。 我几乎完成了,但是当 struct 具有数组字段时,我无法修改它。 我试图用数组值找到合适的函数,但我找不到。 这是我的代码。

edit_struct_via _interface.go

package main

import (
    "reflect"
    "fmt"
)


type myStruct struct {
    Name    [8]byte
    TID     [4]byte
    Chksum  uint16
}

func main() {
    var s = new(myStruct)
    name := []byte("Mike")
    tid  := []byte{0x01, 0x02, 0x03, 0x05}
    setArrayField(name, s.Name[:])
    setArrayField(tid, s.TID[:])
    s.Chksum = 0x0101

    myObj := reflect.ValueOf(s).Elem()

    typeOfMyObj := myObj.Type()
    for i := 0; i < myObj.NumField(); i++ {
        f := myObj.Field(i)
        fmt.Printf("%d: %s %s = %v\n", i,
            typeOfMyObj.Field(i).Name, f.Type(), f.Interface())
    }

    newName := []byte("Emil")
    myObj.Field(0).SetBytes(newName[:])
    // I trying to change array field at here, but panic occur here.
    // Because SetBytes must been called on slice, but myObj.Field(0) is array.

    myObj.Field(2).SetUint(99)             // It works.
    fmt.Println("s is now: ", *s)
}

func setArrayField(src []byte, target []byte) {
    // retun if src is bigger than target
    if len(src) > len(target){
        return
    }

    for index, item := range src {
        target[index] = item
    }
}

当我运行上面的代码时,我得到

0: Name [8]uint8 = [77 105 107 101 0 0 0 0]
1: TID [4]uint8 = [1 2 3 5]
2: Chksum uint16 = 257
panic: reflect: call of reflect.Value.SetBytes on array Value

goroutine 1 [running]:
reflect.flag.mustBe(...)
        /usr/local/go/src/reflect/value.go:208
reflect.Value.SetBytes(0x4a6180, 0xc000098010, 0x191, 0xc000098078, 0x4, 0x4)
        /usr/local/go/src/reflect/value.go:1557 +0x140
main.main()
        /home/hornet/go/src/gateway-golang/edit_struct_via _interface.go:33 +0x522
exit status 2

我使用的是 Ubuntu 18.04 和 go1.13 linux/amd64

SetBytes的文档说:

SetBytes 设置 v 的基础值。 如果 v 的基础值不是一个字节片,它会发生恐慌。

底层值是恐慌消息中所述的数组。

通过将字节复制到数组来修复:

reflect.Copy(myObj.Field(0), reflect.ValueOf(newName))

在操场上运行它

SetBytes适用于切片。 Name不是切片,它是一个数组[8]byte

切片和数组之间存在重要差异,您不能真正将其用于另一个。 切片具有对数组的引用,当您传递切片时,会将引用传递给后备数组。 数组就像一个结构体,当您将数组传递给函数时,您传递的是该数组的副本,而不是对它的引用。

您可以使用 reflect.Copy 来复制字节:

reflect.Copy(myObj.Field(0), reflect.ValueOf(newName))

暂无
暂无

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM