Struct Embedding

Go supports embedding of structs and interfaces to express a more seamless composition of types. This is not to be confused with //go:embed which is a go directive introduced in Go version 1.16+ to embed files and folders into the application binary.

package main

import "fmt"

type base struct {
    num int
}

func (b base) describe() string {
    return fmt.Sprintf("base with num=%v", b.num)
}

// A container embeds a base. An embedding looks like a field without a name.
type container struct {
    base
    str string
}

func main() {

    // When creating structs with literals, we have to initialize
    // the embedding explicitly; here the embedded type serves as the field name.
    co := container{
        base: base{
            num: 1,
        },
        str: "some name",
    }

    // We can access the base’s fields directly on co, e.g. co.num.
    fmt.Printf("co={num: %v, str: %v}\n", co.num, co.str)

    // Alternatively, we can spell out the full path using the embedded type name.
    fmt.Println("also num:", co.base.num)

    // Since container embeds base, the methods of base also
    // become methods of a container.
    // Here we invoke a method that was embedded from base directly on co.
    fmt.Println("describe:", co.describe())

    type describer interface {
        describe() string
    }

    // Embedding structs with methods may be used to
    // bestow interface implementations onto other structs.
    // Here we see that a container now implements
    // the describer interface because it embeds base.
    var d describer = co
    fmt.Println("describer:", d.describe())
}
$ go run struct-embedding.go
co={num: 1, str: some name}
also num: 1
describe: base with num=1
describer: base with num=1
Source | License