我有一个接口Model,该接口由struct实现Person。
Model
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
cannot use newPersons() (type []Person) as type []Model in return argument
我的目标是回到请求任何机型的切片(是否[]Person,[]FutureModel,[]Terminator2000,W / E)。我缺少什么,如何正确实施这样的解决方案?
[]Person
[]FutureModel
[]Terminator2000
简短的答案是您是正确的。结构的片段不等于该结构实现的接口的片段。
A []Person和A []Model具有不同的内存布局。这是因为它们是切片的类型具有不同的内存布局。A Model是接口值,这意味着在内存中它是两个字的大小。一个单词代表类型信息,另一个单词代表数据。A Person是一个结构,其大小取决于它包含的字段。为了从转换[]Person为[]Model,您将需要遍历数组并为每个元素进行类型转换。
[]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