小编典典

切片struct!=它实现的接口切片?

go

我有一个接口Model,该接口由struct实现Person

要获取模型实例,我具有以下辅助函数:

func newModel(c string) Model {
    switch c {
    case "person":
        return newPerson()
    }
    return nil
}

func newPerson() *Person {
    return &Person{}
}

上面的方法允许我返回正确类型的Person实例(以后可以使用相同的方法轻松添加新模型)。

当我尝试执行类似的操作以返回模型切片时,出现错误。码:

func newModels(c string) []Model {
    switch c {
    case "person":
        return newPersons()
    }
    return nil
}

func newPersons() *[]Person {
    var models []Person
    return &models
}

去抱怨: cannot use newPersons() (type []Person) as type []Model in return argument

我的目标是回到请求任何机型的切片(是否[]Person[]FutureModel[]Terminator2000,W /
E)。我缺少什么,如何正确实施这样的解决方案?


阅读 82

收藏
2020-07-02

共1个答案

小编典典

简短的答案是您是正确的。结构的片段不等于该结构实现的接口的片段。

A []Person和A []Model具有不同的内存布局。这是因为它们是切片的类型具有不同的内存布局。A
Model是接口值,这意味着在内存中它是两个字的大小。一个单词代表类型信息,另一个单词代表数据。A
Person是一个结构,其大小取决于它包含的字段。为了从转换[]Person[]Model,您将需要遍历数组并为每个元素进行类型转换。

由于此转换是O(n)操作,并且会导致创建新的切片,因此Go拒绝隐式地执行此操作。您可以使用以下代码明确地做到这一点。

models := make([]Model, len(persons))
for i, v := range persons {
    models[i] = Model(v)
}
return models

就像dskinner指出的那样,您很可能需要一个切片指针而不是一个指向切片的指针。通常不需要指向切片的指针。

*[]Person        // pointer to slice
[]*Person        // slice of pointers
2020-07-02