当前位置: 首页 > 面试题库 >

当参数不是go中的指针时,使用反射通过引用更新值

雍志文
2023-03-14
问题内容

我一直在学习反射,指针和接口的基础知识时遇到困难,所以这是我似乎无法弄清楚的另一个入门级问题。

这段代码做了我想要的事情-我正在使用反射将另一个记录添加到键入为接口的切片中。

package main

import (
  "reflect"
  "log"
)
type Person struct {
  Name string
}
func Add(slice interface{}) {
  s := reflect.ValueOf(slice).Elem()
  // in my actual code, p is declared via the use of reflect.New([Type])
  p := Person{Name:"Sam"}

  s.Set(reflect.Append(s,reflect.ValueOf(p)))
}

func main() {
  p := []Person{}
  Add(&p)
  log.Println(p)
}

如果我将Add和main函数更改为此,那么事情将无法按我想要的方式工作。

func Add(slice interface{}) {
  s := reflect.ValueOf(&slice).Elem()
  p := Person{Name:"Sam"}

  s.Set(reflect.Append(reflect.ValueOf(slice),reflect.ValueOf(p)))
  log.Println(s)
}

func main() {
  p := []Person{}
  Add(p)
  log.Println(p)
}

也就是说,log.Println(p)最后没有Sam像我希望的那样显示带有记录的切片。
所以我的问题是,是否有可能Add()收到不是指针的切片,而我仍然可以编写一些代码Add()来产生第一种情况下显示的结果?

我最近的许多问题都围绕着此类主题展开,因此,我仍然需要花一些时间来弄清楚如何有效地使用反射包。


问题答案:

不,如果不传递指向切片的指针,则不可能在函数中附加切片。这与反射无关,但与变量如何传递到函数有关。这是相同的代码,但修改为不使用反射:

package main

import (
        "log"
)

type Person struct {
        Name string
}

func AddWithPtr(slicep interface{}) {
        sp := slicep.(*[]Person)

        // This modifies p1 itself, since *sp IS p1
        *sp = append(*sp, Person{"Sam"})
}

func Add(slice interface{}) {
        // s is now a copy of p2
        s := slice.([]Person)

        sp := &s

        // This modifies a copy of p2 (i.e. s), not p2 itself
        *sp = append(*sp, Person{"Sam"})
}

func main() {
        p1 := []Person{}
        // This passes a reference to p1
        AddWithPtr(&p1)
        log.Println("Add with pointer:   ", p1)

        p2 := []Person{}
        // This passes a copy of p2
        Add(p2)
        log.Println("Add without pointer:", p2)
}

(上面,当它说“切片”的复制时,这并不意味着基础数据的副本-只是切片)

传递切片时,该函数有效地获得一个新切片,该切片引用与原始数据相同的数据。在函数中附加到切片将增加新切片的长度,但不会更改传入的原始切片的长度。这就是为什么原始切片保持不变的原因。



 类似资料:
  • 问题内容: 我有一个函数,可以遍历作为参数传递的接口的所有字段。为了实现这一点,我正在使用反射。问题是我不知道如何获取非指针字段的地址。这是一个例子: 上面的代码代表了我的测试结构。现在,这是遍历指定结构并列出有关其详细信息的实际函数: 这是结构实例化/初始化之后的实际测试: 最后是InspectStruct调用的输出: 如您所见,我正在使用递归,因此,如果其中一个字段是结构类型,则可以为其调用I

  • Go语言程序中对指针获取反射对象时,可以通过 reflect.Elem() 方法获取这个指针指向的元素类型。这个获取过程被称为取元素,等效于对指针类型变量做了一个 操作,代码如下: 代码输出如下: name: ''  kind: 'ptr' element name: 'cat', element kind: 'struct' 代码说明如下: 第 15 行,创建了cat结构体的实例,ins 是一个

  • 问题内容: 想象以下情况: 我正在专门使用反射来分析MyClass ,在这种情况下,我将获得以下字段(和Types,使用Field的getType()): 我想获取T的实际Type,由于扩展符号中的显式“ String”,因此在运行时就知道了它,我该如何获取myField的非遗传类型? 解决的编辑: 答案似乎是“你做不到”。对于以后可能看这个问题的人,我建议使用Jackson(我试图这样做来生成J

  • 问题内容: 我最近才刚开始接触Go,我有一个主要的困惑点:我在努力理解何时确切需要显式取消引用指针。 例如,我知道运算符将处理对指针的解引用 在其他哪些情况下会这样做?例如,似乎使用数组。 我在规范中找不到此内容,指针部分很短,甚至没有提到取消引用。对取消引用go的指针的规则进行的任何澄清都将很棒! 问题答案: 所述选择器表达式(例如)这是否: 选择器会 自动取消引用 结构的 指针 。如果是指向结

  • 问题内容: 是否可以获取通用参数的类型? 一个例子: 问题答案: 我曾经偶然发现的一种结构看起来像 因此,似乎有些不可思议的想法使我感到遗憾,我很不了解…对不起。

  • 如果反射值对象(reflect.Value)中值的类型为函数时,可以通过 reflect.Value 调用该函数。使用反射调用函数时,需要将参数使用反射值对象的切片 []reflect.Value 构造后传入 Call() 方法中,调用完成时,函数的返回值通过 []reflect.Value 返回。 下面的代码声明一个加法函数,传入两个整型值,返回两个整型值的和。将函数保存到反射值对象(refle