Structure slicing! =The interface slice it implements?

PHPz
Release: 2024-02-09 13:03:08
forward
396 people have browsed it

Structure slicing! =The interface slice it implements?

php editor Apple is here to reveal a puzzle about structure slicing: What is the difference between structure slicing and the interface slicing it implements? In the Go language, slices are a convenient and flexible data structure that can change size dynamically. Struct slicing is a special form of slicing, which stores elements of structure type. But how is it different from a slice that implements the same interface? Let us find out the answer together.

Question content

I have an interface Model, which is implemented by struct Person.

To get the model instance, I have the following helper function:

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

func newPerson() *Person {
    return &Person{}
}
Copy after login

The above method allows me to return a Person instance of the correct type (new models can be easily added later using the same method).

When I try to do something like this to return a model slice, I get an error. Code:

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

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

Go complains: Cannot use newPersons() (type []Person) as a return parameter of type []Model

My goal is to return a slice of whatever model type is requested (whether []Person, []FutureModel, []Terminator2000, w/ e). What am I missing and how to properly implement such a solution?

Solution

This is very similar to the question I just answered: https://www.php.cn/link/2c029952e202c0e560626a4c5980d64c

The short answer is you are right. A slice of a struct is not equal to a slice of the interface implemented by the struct.

[]Person and []Model have different memory layouts. This is because the types they belong to have different memory layouts. Model is an interface value, which means that in memory it is two words in size. One word represents type information and the other word represents data. Person is a structure whose size depends on the fields it contains. In order to convert from []Person to []Model, you need to loop through the array and typecast each element.

Since this conversion is an O(n) operation and results in the creation of a new slice, Go refuses to perform this operation implicitly. You can do this explicitly using the following code.

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

As dskinner pointed out , you most likely need a pointer to a slice, not a pointer to a slice. Pointers to slices are generally not needed.

*[]Person        // pointer to slice
[]*Person        // slice of pointers
Copy after login

The above is the detailed content of Structure slicing! =The interface slice it implements?. For more information, please follow other related articles on the PHP Chinese website!

Related labels:
source:stackoverflow.com
Statement of this Website
The content of this article is voluntarily contributed by netizens, and the copyright belongs to the original author. This site does not assume corresponding legal responsibility. If you find any content suspected of plagiarism or infringement, please contact admin@php.cn
Popular Tutorials
More>
Latest Downloads
More>
Web Effects
Website Source Code
Website Materials
Front End Template
About us Disclaimer Sitemap
php.cn:Public welfare online PHP training,Help PHP learners grow quickly!