一块结构与一段指向结构的指针

ebl*_*ood 3 go

我不明白下面这段代码的行为.在创建一个匹配结构列表作为结构指针切片时,代码总是打印原始数组的最后一个元素(实际上不匹配) - 打印12和12.但是,如果我将匹配更改为[] Widget代替[]*Widget,然后它将打印10和11.

为什么是这样?

package main

import (
    "fmt"
)

func main() {

    type Widget struct {
        id    int
        attrs []string
    }

    widgets := []Widget{
        Widget{
            id:    10,
            attrs: []string{"blah", "foo"},
        },
        Widget{
            id:    11,
            attrs: []string{"foo", "bar"},
        },
        Widget{
            id:    12,
            attrs: []string{"xyz"},
        },
    }

    var matches []*Widget
    for _, w := range widgets {
        for _, a := range w.attrs {
            if a == "foo" {
                matches = append(matches, &w)
                break
            }
        }
    }

    for _, m := range matches {
        fmt.Println(m.id)
    }

}
Run Code Online (Sandbox Code Playgroud)

eug*_*ioy 6

那是因为当你使用指针添加&w到数组时.

请注意,这w实际上是循环中使用的局部变量,因此这不是您要添加到matches数组的地址.

(即使变量的w通过循环改变,它的地址保持不变)

当循环结束时,w以最后一个值结束,这就是它打印12两次的原因.

您需要添加匹配的元素的地址.

如果你这样做:

matches = append(matches, &widgets[i])
Run Code Online (Sandbox Code Playgroud)

然后它也可以用指针工作.

改装Go游乐场供您测试:

https://play.golang.org/p/YE-cokyEHu