Golang使用reflect在struct中设置nil字符串指针值

时间:2017-05-02 17:51:54

标签: pointers go reflection struct null

使用Go的反射包,如果指针为零,有没有办法在结构中设置指针?查看反馈包,如果reflect.Value.CanSet()false,那么任何Set()次调用都会产生恐慌。还有另外一种方法,只使用反射包,而不依赖于对结构的任何直接引用?例如,如何在下面的代码中将空的姓氏设置为“Johnson”?

package main

import (
    "fmt"
    "reflect"
)

type Person struct {
    FirstName *string
    LastName  *string
}

func main() {
    p := Person{}
    firstName := "Ryan"
    p.FirstName = &firstName

    rp := reflect.ValueOf(p)
    fmt.Println("rp:", rp)

    for i := 0; i < rp.NumField(); i++ {
        fmt.Printf("%s: Pointer: %d CanSet: %t\n", rp.Type().Field(i).Name, rp.Field(i).Pointer(), rp.Field(i).Elem().CanSet())
    }

    rp.Field(0).Elem().SetString("Brian")
    fmt.Println(*p.FirstName)

    // Yields nil pointer error
    // rp.Field(1).Elem().SetString("Johnson")
    // fmt.Println(*p.LastName)

    fmt.Println(rp.Field(1).Type()) 
    fmt.Println(rp.Field(1).CanSet())
    // fmt.Println(rp.Field(1).Elem().Type()) // nil pointer here also
    fmt.Println(rp.Field(1).Elem().CanSet())
}

See in Golang Playground

1 个答案:

答案 0 :(得分:5)

首先需要指向Person{}的指针,因为您需要为LastName字段设置一个值。

p := &Person{}

然后你可以设置一个有效的指针值到LastName字段,这将允许你设置字符串值:

rp.Field(1).Set(reflect.New(rp.Field(1).Type().Elem()))
rp.Field(1).Elem().SetString("Jones")

https://play.golang.org/p/f5MjpkDI2H