[英]Using reflect to update value by reference when argument is not a pointer in go
我一直在學習反射,指針和接口的基礎知識時遇到困難,所以這是我似乎無法弄清楚的另一個入門級問題。
這段代碼完成了我想要的工作-我正在使用反射將另一個記錄添加到鍵入為接口的切片中。
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)
}
(上面,當它說“切片”的復制時,並不意味着底層數據的復制-只是切片)
當您傳遞切片時,該函數有效地獲得一個新切片,該切片引用與原始數據相同的數據。 在函數中附加到切片將增加新切片的長度,但不會更改傳入的原始切片的長度。這就是為什么原始切片保持不變的原因。
聲明:本站的技術帖子網頁,遵循CC BY-SA 4.0協議,如果您需要轉載,請注明本站網址或者原文地址。任何問題請咨詢:yoyou2525@163.com.