如何使用反射将值设置为包含 nil 的指针

Som*_*ent 3 reflection go

我正在尝试将一个值设置为像这样的结构中的 nil 指针。

// https://play.golang.org/p/jPTMNC_ZQ9
package main

import (
    "fmt"
    "reflect"
)

type T struct {
    A *int
}

func main() {
    fmt.Println("Hello, playground")
    t := &T{}

    v := 1
    vptr := &v

    CopyValue(vptr, t.A) // I want to set t.A to contain 1
}

func CopyValue(src interface{}, dest interface{}) {
    srcRef := reflect.ValueOf(src)
    if srcRef.Kind() == reflect.Ptr {
        srcRef = srcRef.Elem()
    }

    destRef := reflect.New(srcRef.Type()).Elem()
    destRef.Set(srcRef)
    reflect.ValueOf(dest).Elem().Set(destRef)
}
Run Code Online (Sandbox Code Playgroud)

但是,我遇到以下错误:

panic: reflect: call of reflect.Value.Set on zero Value

goroutine 1 [running]:
reflect.flag.mustBeAssignable(0x0, 0x1040a128)
    /usr/local/go/src/reflect/value.go:221 +0x260
reflect.Value.Set(0x0, 0x0, 0x0, 0xdefc0, 0x1040a128, 0x182)
    /usr/local/go/src/reflect/value.go:1339 +0x40
main.CopyValue(0xd7860, 0x1040a124, 0xd7860, 0x0)
    /tmp/sandbox487854080/main.go:30 +0x1a0
main.main()
    /tmp/sandbox487854080/main.go:19 +0x100
Run Code Online (Sandbox Code Playgroud)

我究竟做错了什么?

eug*_*ioy 5

为了能够修改t.A指向的内容,您需要将对它的引用发送到您的CopyValue函数。

CopyValue(vptr, &t.A) // (note the &)
Run Code Online (Sandbox Code Playgroud)

然后您可以将指针分配给新地址:

func CopyValue(src interface{}, dest interface{}) {
    srcRef := reflect.ValueOf(src)
    vp := reflect.ValueOf(dest)
    vp.Elem().Set(srcRef)
}
Run Code Online (Sandbox Code Playgroud)

请参阅此处的第三个“反射定律”:https://blog.golang.org/laws-of-reflection

完整的工作代码:

package main

import (
    "fmt"
    "reflect"
)

type T struct {
    A *int
}

func main() {
    t := &T{}
    v := 1
    vptr := &v
    CopyValue(vptr, &t.A) // we pass a reference to t.A since we want to modify it
    fmt.Printf("%v\n", *t.A)
}

func CopyValue(src interface{}, dest interface{}) {
    srcRef := reflect.ValueOf(src)
    vp := reflect.ValueOf(dest)
    vp.Elem().Set(srcRef)
}
Run Code Online (Sandbox Code Playgroud)