嵌套切片未更新

  • 本文关键字:更新 切片 嵌套 go
  • 更新时间 :
  • 英文 :


我学习围棋大约1个月了,我读了很多关于rangeslices的文章,以及这个复杂主题带来的所有危险。不幸的是,我在这个特殊的问题上有点不知所措。我希望这个例子不要太做作,但我试着想出一个最小的例子。

问题是,当我第一次通过addNoise添加bird时,会添加bird,但它不会发出声音。如果我用bird第二次调用addNoise,那么它似乎可以正常工作。然而,我仍然错过了我试图添加的第一个声音。

我认为问题完全在于我如何在getOrCreateAnimal中返回animal,但我不认为我做错了什么。我怀疑新创建的animal的返回地址与存储在切片中的地址不同。

我可能忽略了关于切片的一个重要事实,我真的很感激有人帮助我理解我错过了什么。

我在Go 1.15上运行这个例子。

package main
import "fmt"
type Barn struct {
animals []Animal
}
type Animal struct {
name   string
noises []string
}
type BarnService struct {
barn Barn
}
func (fs *BarnService) indexOfAnimal(name string) int {
for idx := range fs.barn.animals {
if fs.barn.animals[idx].name == name {
return idx
}
}
return -1
}
func (fs *BarnService) getAnimal(name string) *Animal {
idx := fs.indexOfAnimal(name)
if idx == -1 {
return nil
}
return &fs.barn.animals[idx]
}
func (fs *BarnService) getOrCreateAnimal(name string) (animal *Animal) {
animal = fs.getAnimal(name)
if animal == nil {
animal = &Animal{name: name, noises: []string{}}
fs.barn.animals = append(fs.barn.animals, *animal)
}
return
}
func (fs *BarnService) addNoise(noise string, barValue string) {
animal := fs.getOrCreateAnimal(barValue)
animal.noises = append(animal.noises, noise)
}
var fs = BarnService{
barn: Barn{animals: []Animal{{name: "cat"}, {name: "cow"}}},
}
func main() {
fmt.Printf("1-barn = %+vn", fs.barn)
fs.addNoise("meow", "cat")
fs.addNoise("purr", "cat")
fs.addNoise("hiss", "cat")
fs.addNoise("moo", "cow")
fs.addNoise("moo", "cow")
fs.addNoise("peep", "bird")  //Bird gets created but doesn't make a peep
fmt.Printf("2-barn = %+vn", fs.barn)
fs.addNoise("peep", "bird")  //OK now bird peeps, but it should say "peep peep"
fmt.Printf("3-barn = %+vn", fs.barn)
}

输出:

1-barn = {animals:[{name:cat noises:[]} {name:cow noises:[]}]}
2-barn = {animals:[{name:cat noises:[meow purr hiss]} {name:cow noises:[moo moo]} {name:bird noises:[]}]}
3-barn = {animals:[{name:cat noises:[meow purr hiss]} {name:cow noises:[moo moo]} {name:bird noises:[peep]}]}

返回的Animal实例和添加到切片中的实例不相同。尝试返回一个指向切片中实例的指针,您应该可以完成

func (fs *BarnService) getOrCreateAnimal(name string) (animal *Animal) {
animal = fs.getAnimal(name)
if animal == nil {
ani := Animal{name: name, noises: []string{}}
fs.barn.animals = append(fs.barn.animals, ani)
animal = &fs.barn.animals[len(fs.barn.animals)-1]
}
return
}

去游乐场

最新更新