Interfaces

pavan-nambi's avatar
Published Apr 25, 2023
Contribute to Docs

An interface is composed of set of method signatures. These method signatures define the input and return values of which a data type or struct can conform to. In order to implement an interface, the type or struct must satisfy all the defined method signatures.

Creating an Interface

type InterfaceName interface {
  // Method signatures
}

Note: Interface names are written in either camelCase or PascalCase. Conventionally, interfaces with one method should end with -er.

Example

package main
import (
"fmt"
)
type marriage interface {
love()
arranged()
}
type john struct{}
func (a *john) love() {
fmt.Println("love with john")
}
func (a *john) arranged() {
fmt.Println("arranged marriage with john")
}
func newLife(guy marriage) {
guy.love()
guy.arranged()
}
func main() {
newLife(&john{})
}

Output:

love with john
arranged marriage with john

In the above example:

  • The marriage interface has two methods: love and arranged.
  • john implements the marriage interface but does not have any explicit declaration to implement the interface.
  • As long as the struct has all the methods in the interface, it is said to implement the interface.
  • The function newLife() takes a marriage interface as an argument.

One benefit of using an interface is that any struct that implements the marriage interface can be passed as an argument to the newLife() function. This means that john can have its love and arranged methods modified without affecting newLife(). As long as the marriage interface is still implemented, any modified structure can still be used with the newLife() function.

Go’s compiler will ensure that all the methods in the interface are implemented in the struct.

A type implements an interface by implementing its methods, In Go, interfaces are implemented implicitly and enables polymorphism.

Empty Interface

An interface with no methods is called an empty interface.

type interfaceName interface {}

The tradeoff of using an empty interface is the loss of type safety. Since there are no methods, it’s possible to pass any type to a function that takes an empty interface as an argument.

Example

The Println() function in the fmt package takes an empty interface as an argument.

Below is the Go source code for the fmt.Println() function:

// Println formats using the default formats for its operands and writes to standard output.
// Spaces are always added between operands and a newline is appended.
// It returns the number of bytes written and any write error encountered.
func Println(a ...interface{}) (n int, err error) {
return Fprintln(os.Stdout, a...)
}

Note: Due to the empty interface, any type can be passed to the Println() function.

Embedding an Interface

An interface in go can be embedded in other interfaces or structs.

package main
import "fmt"
// Any type that needs to implement marriage should also implement all methods of dating.
type marriage interface {
dating
love()
arranged()
}
type dating interface {
compatibility()
}
type john struct {
age int
}
func (a john) love() {
fmt.Println("love with john")
}
func (a john) arranged() {
fmt.Println("arranged marriage with john")
}
func (a john) compatibility() {
fmt.Println("compatible with john")
}
type parents struct {
//Named marriage interface
m marriage
quote string
}
type lovers struct {
// Unnamed marriage interface
marriage
quote string
}
func main() {
a := john{age: 24}
p := parents{quote: "I saw that you were perfect, and so I loved you. Then I saw that you were not perfect and I loved you even more", m: a}
// p.love()
// As this has named marriage interface m calling disquarely with .m would raise compiler error
/*-----------------------ERROR-----*---------------------------*/
// ./prog.go:39:4: p.love undefined (type parents has no field or method love)
p.m.love()
p.m.compatibility()
p.m.arranged()
l := lovers{quote: "I saw that you were perfect, and so I loved you. Then I saw that you were not perfect and I loved you even more", marriage: a}
fmt.Println("\n", l.quote, "\n")
// It's possible to disquarely access the methods of embedded interface if the embedded interface is anonymous or unnamed.
l.love()
l.compatibility()
l.arranged()
}

This example results in the following output:

love with john
compatible with john
arranged marriage with john
I saw that you were perfect, and so I loved you. Then I saw that you were not perfect and I loved you even more
love with john
compatible with john
arranged marriage with john
Program exited.

Codebyte Example

A simple example on how to use interfaces, square and circle both implement the geometry interface as both implement the required methods.

In the code below, the geometry interface has two methods area() and perimeter(), both of which return a float64. square and circle implement these methods, allowing them to be used interchangeably as geometric objects.

The code then defines a printGeometry() function, which takes a geometry object as its parameter and prints its details, including its area and perimeter.

In the main() function, a square and a circle are created, and the printGeometry() function is called for each of them, demonstrating how the two different types can be used interchangeably as geometric objects.

Code
Output
Loading...

All contributors

Contribute to Docs

Learn Go on Codecademy