Understanding Go’s Type System: Interfaces and Embedding Demystified

Go is a statically typed language, which means that the type of a value is determined at compile time. Go’s type system is simple and straightforward, making it easy for developers to write correct and maintainable code. But, as with any language, there are some concepts that can take a bit of time to fully grasp. In this article, we’ll be focusing on two of these concepts in Go’s type system: interfaces and embedding.

What’s the Deal with Interfaces in Go?

Interfaces in Go are a way to define a set of methods that a type must implement in order to be considered of that interface type. In simpler terms, interfaces provide a contract that types must adhere to. This makes it possible for types to be interchangeable and reduces coupling between different parts of a program.

For instance, consider the following interface:

type Reader interface { Read(p []byte) (n int, err error) }
Code language: Go (go)

This interface defines a single method, Read, which takes a byte slice p and returns the number of bytes read and an error (if any). Any type that implements this method can be considered a Reader. This includes types such as os.File, bytes.Buffer, and many others.

func ReadFrom(r Reader, p []byte) (int, error) { return r.Read(p) }
Code language: Go (go)

The ReadFrom function takes a Reader and a byte slice, and returns the number of bytes read and an error. Since any type that implements the Reader interface can be passed to this function, it can be used with a variety of types.

Interfaces in Go are a game-changer! By defining an interface, you can write code that can work with any type that implements that interface, rather than being tied to a specific type. This allows for a high degree of code reuse and flexibility.

Embedding in Go: A Powerful Tool

Embedding in Go is a way to reuse code by including one type as a field within another type. The embedded type acts as an anonymous field within the outer type, and its methods are promoted to the outer type. This allows for code reuse without inheritance or the use of composition.

For example, consider the following code:

type User struct { Name string Email string } func (u *User) String() string { return u.Name + " <" + u.Email + ">" } type Admin struct { User Level string }
Code language: Go (go)

The User type has a String method that returns a string representation of the user. The Admin type embeds the User type as an anonymous field, and its methods are promoted to the Admin type.

func main() { a := Admin{ User: User{ Name: "Mehul Mohan", Email: "[email protected]", }, Level: "admin", } fmt.Println(a.String()) }
Code language: Go (go)

In this code, we create an Admin value and call its String method. Since the Admin type embeds the User type, it has access to the String method defined on User.

This allows us to reuse the String method without having to duplicate the code. It’s like having a trusty sidekick, always there to help you out!

Embedding is a fantastic feature in Go and can help you write more modular and reusable code. By embedding types, you can break down complex types into smaller, more manageable pieces. Additionally, embedding can make testing easier, as you can test the behavior of individual embedded types without having to test the entire outer type. It’s like having a multi-tool in your toolbox – you always have the right tool for the job!

When to Use Interfaces and Embedding

So, when should you use interfaces and embedding in Go?

Interfaces are the way to go when you need to define a common behavior that multiple types can implement. This can be useful for writing generic functions or creating a more flexible codebase. For example, you can define a Reader interface that multiple types can implement, allowing you to write a generic ReadFrom function that can work with any type that implements Reader.

Embedding is the perfect solution when you want to reuse code within a type. This can help reduce code duplication and break down complex types into smaller, more manageable pieces. For instance, you can embed a User type within an Admin type to reuse the code for representing a user.


In conclusion, both interfaces and embedding play a crucial role in Go’s type system. By using interfaces, you can define a common behavior that multiple types can implement. And by using embedding, you can reuse code within a type. Both of these concepts can help you write more modular, reusable, and maintainable code.

I hope this article helped demystify Go’s type system and the concepts of interfaces and embedding for you. Let’s code and have some fun! 🚀

Sharing is caring

Did you like what Mehul Mohan wrote? Thank them for their work by sharing it on social media.


No comments so far