Go Interfaces: Difference between revisions

From NovaOrdis Knowledge Base
Jump to navigation Jump to search
 
(111 intermediate revisions by the same user not shown)
Line 5: Line 5:
* [[Go Language]]
* [[Go Language]]
* [[Go_Language_Object_Oriented_Programming#Interfaces|Object-Oriented Programming in Go]]
* [[Go_Language_Object_Oriented_Programming#Interfaces|Object-Oriented Programming in Go]]
* [[Go_Methods#Overview|Methods]]


=Overview=
=Overview=


An '''interface''' is a set of method signatures. The interface is declared with the keyword <code>[[Go_Language#type_keyword|type]]</code>, followed by the name of the interface, followed by the keyword <code>[[Go_Language#interface_keyword|interface]]</code> and by the method signatures enclosed in a curly brace block. Each signature contains the method name, optional parameter names and mandatory parameter types, and optional return variable names and mandatory return types. Providing parameter and return names is recommended, as it arguably exposes more semantics, making the code clearer. See [[#Declaration|example]], below. There are no method implementations.  
An '''interface''' is a set of method signatures. The interface is declared with the keyword <code>[[Go_Language#type_keyword|type]]</code>, followed by the name of the interface, followed by the keyword <code>[[Go_Language#interface_keyword|interface]]</code> and by the method signatures enclosed in a curly brace block. Each signature contains the method name, optional parameter names and mandatory parameter types, and optional return variable names and mandatory return types. Providing parameter and return names is recommended, as it arguably exposes more semantics, making the code clearer. See [[#Declaration|example]], below.  
 
The interface declaration contains no method implementations.  


Interfaces are not fully-featured types, they're less than that, but they're used to express conceptual similarities between types. If several types implement the same interface, they are similar in a way that is important to the application. An interface highlights the similarities between the types implementing it and conceals the differences.
Interfaces are not fully-featured types, they're less than that, but they're used to express conceptual similarities between types. If several types implement the same interface, they are similar in a way that is important to the application. An interface highlights the similarities between the types implementing it and conceals the differences.


Interfaces are not declared to be satisfied, they are satisfied implicitly.  A type '''implements''' (or '''satisfies''') an interface if the type defines '''all''' methods specified in the interface. Of course, the type can have other methods not specified by the interface.
<span id='accept_interfaces_return_structs'></span>An interface does need to be formally declared on a target type for that type to implement it. Interfaces are satisfied implicitly.  A type '''implements''' or '''satisfies''' an interface if the type defines '''all''' methods specified in the interface. Of course, the type can have other methods not specified by the interface. This is called <span id='Duck_Typing'></span>[[Programming_Languages_Concepts#Duck_Typing|duck typing]]. If a type implements an interface by the virtue of duck typing, a value of the type can be stored in a value of that interface type and the compiler ensure the assignments are correct at compile-time. A type can implement multiple interfaces.
 
Interfaces can be thought of a kind of conceptual inheritance: types inherit behavior specified by an interface. Interfaces help supporting polymorphism in Go: different types implement different behaviors specified by the '''same''' interface. For more details see: {{Internal|Go_Inheritance_and_Polymorphism#Overview|Go Inheritance and Polymorphism}}
 
Go interfaces allow programs to model behavior rather than model types. The behavior is "attached" to existing types, via methods, and independently, the behavior is given a name, the name of the interface that groups together the methods. Go type system is structural, not nominal. The interfaces are behavior contracts, and they belong in the consuming code, not in the producing code. '''A common pattern should be to accept interfaces and return structs'''.  


Interfaces can be thought of a kind of conceptual inheritance (types inherit behavior specified by an interface) and help with implementing [[Go_Language_Object_Oriented_Programming#Polymorphism|polymorphism]] in Go.
:::<font color=darkkhaki>This is debatable: for an interface itself, it feels natural that the methods should return the same interface, not an implementing type. What if the implementing type is package private, and the interface is public?</font>
 
<font color=darkkhaki>A type and its associated [[Go_Concepts_-_The_Type_System#Pointer_Types|pointer type]] may or may not implement the same interface, depending on how the methods are associated with the value and pointer receivers of that type.</font>


"The bigger the interface, the weaker the abstraction." The most important thing about interfaces in Go is that the smaller the interface is, the more useful it is. This is a Go-specific idea: we want to make little interfaces, so we can build components that share them.
"The bigger the interface, the weaker the abstraction." The most important thing about interfaces in Go is that the smaller the interface is, the more useful it is. This is a Go-specific idea: we want to make little interfaces, so we can build components that share them.


Go type system is structural,  not nominal. The interfaces are behavior contracts, and they belong in the consuming code, not in the producing code. A common pattern should be to accept interfaces and return structs.
Interfaces are defined at the "fault lines" of the system's architecture. These are between the <code>main()</code> function and the rest, and along the package API boundaries.  


Interfaces are defined at the "fault lines" of the system's architecture. These are between the <code>main()</code> function and the rest, and along the package API boundaries. Interfaces should not be defined before a realistic example of usage exists.
Interfaces should not be defined before a realistic example of usage exists.
 
Interfaces are rigid. For a library that exposes interfaces in the public API, any change to the interface requires releasing a new major version, since it breaks all third-party implementations.
:::<font color=darkkhaki>Is that right? What if we '''add''' to the interface?</font>


=Declaration=
=Declaration=
Line 26: Line 38:
The following syntax declares an interface:
The following syntax declares an interface:


<syntaxhighlight lang='go'>
<font size=-1.5>
type <Interface_Name> interface {
<font color=green><b>type</b></font> <I><InterfaceName></I> <font color=green><b>interface</b></font> {
  <method_signature_1>
  <font color=teal>// Interface method set:</font>
  <method_signature_2>
  <I><method_signature_1></I>
  ...
  <I><method_signature_2></I>
}
  ...
</syntaxhighlight>
}
</font>
Example:
Example:
<syntaxhighlight lang='go'>
<syntaxhighlight lang='go'>
type SomeInterface interface {
type SomeInterface interface {
   MethodA(int) (int, error)                          // No parameter and return value names are specified.
   MethodA(int) (int, error)                          // No parameter and return value names are specified.
   MethodB(someValue int) (someResult int, err error) // Parameter and return value names are provided.
   MethodB(someValue int) (someResult int, err error) // Parameter and return value names are provided.
Line 44: Line 56:


What results after such a declaration is an <span id='Interface_Type'></span>'''interface type'''. For a discussion of concrete type vs. interface types, see: {{Internal|Go_Language#Concrete_vs._Interface_Types|Go Language &#124; Concrete vs. Interface Types}}
What results after such a declaration is an <span id='Interface_Type'></span>'''interface type'''. For a discussion of concrete type vs. interface types, see: {{Internal|Go_Language#Concrete_vs._Interface_Types|Go Language &#124; Concrete vs. Interface Types}}
==Interface Method Set==
An interface method set the list of method declaration specified in the [[#Declaration|interface type declaration]].  The interface type declaration has the only purpose of defining the method set and giving it a name, which is the interface name. Also see: {{Internal|Go_Methods#Method_Set|Method Sets}}


=Idiomatic Interface Conventions=
=<span id='Idiomatic_Interface_Conventions'></span>Naming=
Single-method interfaces are named using the method name plus an "er" suffix. An interface with just one <code>Write</code> method would be called <code>Writer</code>.  
Single-method interfaces are named using the method name plus an "er" suffix. An interface with just one <code>Write</code> method would be called <code>Writer</code>. Sometimes, the result isn't correct English, but we do it anyway. In other cases, we use English to make the result nicer:
<syntaxhighlight lang='go'>
type ByteReader interface {
    ReadByte() (c byte, err error)
}
</syntaxhighlight>
When an interface includes multiple methods, choose a name that accurately describes its purpose.  


Also see: {{Internal|Go_Style#Naming|Go Style}}
Also see: {{Internal|Go_Style#Naming|Go Style}}
Line 55: Line 75:
type SomeImplementation { ... } // SomeInterface is specified nowhere in the SomeImplementation declaration
type SomeImplementation { ... } // SomeInterface is specified nowhere in the SomeImplementation declaration


// Declaring the type as pointer receiver for *all* methods declared in the interface makes the type to implement that interface  
// Declaring the type as pointer receiver for *all* methods declared in the interface makes  
// Value and pointer receivers must not be mixed by the method implementations of an interface
// the type to implement that interface . Value and pointer receivers must not be mixed by  
// the method implementations of an interface.
func (s *SomeImplementation) MethodA(i int) (int, error) { ... }
func (s *SomeImplementation) MethodA(i int) (int, error) { ... }
func (s *SomeImplementation) MethodB(someValue int) (int, error) { ... }
func (s *SomeImplementation) MethodB(someValue int) (int, error) { ... }
</syntaxhighlight>
</syntaxhighlight>
The method signatures from the interface method set are independent on whether the corresponding methods from the concrete types are declared with value or pointer receivers.
==When Does a Type/Pointer Type Implement an Interface?==
Note that notion of a type implementing an interface and its corresponding pointer type implementing the same interface both make sense, and they are subtly different. According to the language specification, a user-defined type <code>T</code> implements an interface if its [[Go_Language_Object_Oriented_Programming#Type_Method_Set|method set]], which are all methods declared with '''a value receiver''' of type <code>T</code>, is a superset of the interface. By extrapolation, a pointer type implements an interface when the method set associated with pointers to that type is a superset of the [[#Interface|interface]]. As explained [[Go_Language_Object_Oriented_Programming#Pointer_Type_Method_Set|here]], the method set associated with the pointer of a type includes implicitly the method set associated with the value of the type.
==What Can Implement an Interface?==
Structs can implement interfaces.
Type aliases can implement interfaces. In general, [[Go_Language#Named_Type|named types]] can implement interfaces, like in the following example where a <code>bool</code> can implement an interface as long as it is named:
<syntaxhighlight lang='go'>
type SomeInterface interface {
    SomeMethod()
}
type someImpl bool
func (s someImpl) SomeMethod() {
fmt.Printf("SomeMethod\n")
}
someImpl(true).SomeMethod()
</syntaxhighlight>
<font color=darkkhaki>What else? Can a function be used as an interface?</font>


=Interface Values=
=Interface Values=
Declaring a variable of an interface type creates an '''interface value'''.
Declaring a variable of an interface type creates an '''interface value''':
 
The interface value is a pair that contains a '''dynamic type''' and a '''dynamic value'''. The dynamic type is a [[Go_Language#Concrete_Type|concrete type]] that implements that interface. The dynamic value is an instance of the dynamic type:
<syntaxhighlight lang='go'>
<syntaxhighlight lang='go'>
var i SomeInterface // i is a pair: (dynamic_type, dynamic_value)
var i SomeInterface // i is a pair: (dynamic_type, dynamic_value)
</syntaxhighlight>
</syntaxhighlight>
The interface value contains a data pair, including a <span id='Dynamic_Type'></span>'''dynamic type''' and a <span id='Dynamic_Value'></span>'''dynamic value'''. The dynamic type, when set, is a [[Go_Language#Concrete_Type|concrete type]] that implements that interface. The dynamic value, when set, is an instance of the dynamic type. Internally, an interface variable is a two-word data structure. The first word contains a pointer to an internal table called <code>iTable</code>, which contains type information about the stored value: the concrete type of the value that has been stored and a list of methods associated with the value. The second word is a reference to the stored value. The combination of type information and pointer binds the relationship between the two values.


In the case of the above declaration, both elements of the pair are <code>nil</code>:
<syntaxhighlight lang='go'>
fmt.Printf("%v\n", i) // prints <nil>
</syntaxhighlight>
Assuming the following interface and concrete type declarations:
Assuming the following interface and concrete type declarations:
<syntaxhighlight lang='go'>
<syntaxhighlight lang='go'>
type SomeInterface interface {
type SomeInterface interface {
   MethodA()
   SomeMethod()
}
}


Line 79: Line 128:
}
}


// MethodA makes SomeImplementation implement the interface.
// SomeMethod makes SomeImplementation implement SomeInterface.
func (s *SomeImplementation) MethodA() {
func (s *SomeImplementation) SomeMethod() {
   if s != nil {
   if s == nil {
     fmt.Printf("data: %s\n", s.data)
     fmt.Println("invocation with a nil dynamic value")
   } else {
   } else {
     fmt.Println("invocation with a nil dynamic value")
     fmt.Printf("data: %s\n", s.data)  
   }
   }
}</syntaxhighlight>
}
... the following variable declarations and initializations create an interface variable and a concrete type variable. The code then assigns the interface variable with the concrete type variable, which will make the interface value get the pair of a dynamic type (<code> SomeImplementation </code>) and the dynamic value (<code>t</code>):
</syntaxhighlight>
... the following variable declaration and initialization ends up in an interface variable whose both dynamic type and dynamic value are set to <code>*SomeImplementation</code> and <code>&SomeImplementation{"test"}</code>, respectively:
<syntaxhighlight lang='go'>
<syntaxhighlight lang='go'>
var i SomeInterface
var i SomeInterface
var t = &SomeImplementation{"blue"}
i = &SomeImplementation{"test"}
i = t
</syntaxhighlight>
</syntaxhighlight>
The interface value can be then used to invoke the method on the dynamic concrete type instance (the dynamic value).
The interface value can be then used to invoke the <code>SomeMethod()</code> method on the dynamic value, which was assigned to the dynamic concrete type instance pointer.
<syntaxhighlight lang='go'>
<syntaxhighlight lang='go'>
i.MethodA() // prints "data: blue"
i.SomeMethod() // prints "data: test"
</syntaxhighlight>
</syntaxhighlight>
An interface value can have a <code>nil</code> dynamic value, and still be usable, as long it is assigned a dynamic type with this syntax:
An interface value can have a <code>nil</code> dynamic value, and still be usable, as long it is assigned a dynamic type with this syntax:
<syntaxhighlight lang='go'>
<syntaxhighlight lang='go'>
var i SomeInterface
var i SomeInterface
var t *SomeType // Declares "t" a pointer to the concrete type, not the concrete type
var t *SomeImplementation
i = t
i = t
</syntaxhighlight>
</syntaxhighlight>
An interface value assigned with a pointer to the dynamic type but not dynamic value can still be invoked into, given that the method implementation cases for the situation when the receiver type [[Go_Language_Object_Oriented_Programming#Implicit_Argument|implicit argument]] is <code>nil</code>. In the example above, we assigned <code>i</code> a dynamic type, <code>SomeImplementation</code>, but not a dynamic value, as <code>t</code> has no concrete value yet. The dynamic type is enough information to go find which implementation <code>MethodA()</code> should use. That is why it is a good idea to guard against <code>nil</code> implicit argument in all concrete methods provided by types that implement interfaces. This way you have a static (in the Java or Python sense of the world) method and an instance method at the same time.
An interface value assigned with a pointer to the dynamic type, but not with a dynamic value can still be invoked into, given that the method implementation cases for the situation when the receiver type [[Go_Language_Object_Oriented_Programming#Implicit_Argument|implicit argument]] is <code>nil</code>. In the example above, we assigned <code>i</code> a dynamic type <code>*SomeImplementation</code> but not a dynamic value, as <code>t</code> has no concrete value yet. The dynamic type is enough information to go find which implementation <code>SomeMethod()</code> should use.  
 
{{Note|This is why it is a good idea to guard against <code>nil</code> implicit argument in all concrete methods provided by types that implement interfaces.}}
 
This way we get a static method, in the Java or Python sense of the word and an instance method at the same time.
<syntaxhighlight lang='go'>
<syntaxhighlight lang='go'>
i.MethodA() // prints "invocation with a nil dynamic value"
i.SomeMethod() // prints "invocation with a nil dynamic value"
</syntaxhighlight>
</syntaxhighlight>
Having a dynamic type but no dynamic value is a legal state. It is a way to implement static (type) methods.
An interface value that carries a dynamic type but no dynamic value is a legal state. It is a way to implement static (type) methods.
 
In the case of an interface variable with <code>nil</code> dynamic value, <code>i == nil</code> evaluates to <code>false</code> but <code>testify</code> <code>require.Nil()</code> evaluates to <code>true</code>.
 
<span id='Nil_Dynamic_Value_Testing_on_Private_Types'></span>This is how to test public interfaces whose implementing type is package-private: {{Internal|Go_Nil_Dynamic_Value_Testing_on_Private_Types#Overview|Testing Public Interfaces whose Implementing Type is Package-Private}}
 
==<tt>nil</tt> Interface Value==
==<tt>nil</tt> Interface Value==
This describes an interface value with a <code>nil</code> dynamic type. In this situation, you cannot call the methods on that interface, because without a dynamic type you can't know which method you are referring to.
This describes an interface value with a <code>nil</code> dynamic type. In this situation, you cannot call the methods on that interface, because without a dynamic type you can't know which method you are referring to.
=Interfaces as Function Parameters and Return Values=
==<span id='Type_Assertions'></span>Type Assertion==
There is syntax in the language which, given an interface variable, allows us to extract the [[#Dynamic_Value|dynamic value]] with the actual [[#Dynamic_Type|dynamic type]]: {{Internal|Go_Type_Assertion#Overview|Type Assertion}}
 
==Type Switch==
{{Internal|Go Type Switch#Overview|Type Switch}}
 
=<span id='Interfaces_as_Function_Parameters_and_Return_Values'></span>Interfaces as Function Parameters and Result Values=
==Interface as Function Parameter==
==Interface as Function Parameter==
When a function is declared with a parameter of an interface type, the function can be invoked with either a value or a pointer of the implementing type.  
Interfaces can be used as arguments to functions. Passing an interface instance insures that the function body can rely on the fact the interface methods are available on the passed instance. When a function is declared with an interface type parameter, the function can be invoked with either a value or a pointer of the implementing type, as long the implementation uses a value receiver type. For the following interface and type declaration:
 
For the following interface and type declaration:
<syntaxhighlight lang='go'>
<syntaxhighlight lang='go'>
type SomeInterface interface {
type SomeInterface interface {
Line 124: Line 187:
}
}
</syntaxhighlight>
</syntaxhighlight>
the type may chose to implement the interface by declaring a value receiver:
the type may chose to implement the interface by using a value receiver:
<syntaxhighlight lang='go'>
<syntaxhighlight lang='go'>
func (s SomeImplementation) MethodA() {
func (s SomeImplementation) MethodA() {
Line 131: Line 194:
</syntaxhighlight>
</syntaxhighlight>


In this case, if a function was declared with a parameter of <code>SomeInterface</code> interface type, the function can be invoked by passing either a value or a pointer to the type instance:
In this case, if a function is declared with a parameter of <code>SomeInterface</code> interface type, the function can be invoked by passing either a value or a pointer to the type instance:


<syntaxhighlight lang='go'>
<syntaxhighlight lang='go'>
Line 148: Line 211:
</syntaxhighlight>
</syntaxhighlight>


If the implementing method uses a pointer receiver:
If the implementation of the interface is done with a pointer receiver:
<syntaxhighlight lang='go'>
<syntaxhighlight lang='go'>
func (s *SomeImplementation) MethodA() {
func (s *SomeImplementation) MethodA() {
fmt.Printf("data: %s\n", s.data)
    fmt.Printf("data: %s\n", s.data)
}
}
</syntaxhighlight>
</syntaxhighlight>


then only a pointer to the implementing type can be passed to the function <code>SomeFunc()</code>:
then only a pointer to the implementing type instance can be passed to the function <code>SomeFunc()</code>:


<syntaxhighlight lang='go'>
<syntaxhighlight lang='go'>
Line 162: Line 225:
</syntaxhighlight>
</syntaxhighlight>


If we attempt to pass an implementing type value, the compiler complains:  
If we attempt to pass an implementing type instance value, the compiler complains:  
<font size=-2>
<font size=-2>
  Cannot use t (variable of type SomeImplementation) as SomeInterface value in argument to SomeFunc: SomeImplementation does not implement SomeInterface (method MethodA has pointer receiver)
  Cannot use t (variable of type SomeImplementation) as SomeInterface value in argument to SomeFunc: SomeImplementation does not implement SomeInterface (method MethodA has pointer receiver)
Line 168: Line 231:


==Interface as Function Result==
==Interface as Function Result==
Functions can return interfaces as results:
Functions can return interfaces as results, as shown below. However, in general, functions [[#accept_interfaces_return_structs |should accept interfaces and return structs]].
 
<syntaxhighlight lang='go'>
<syntaxhighlight lang='go'>
type SomeInterface interface {
type SomeInterface interface {
Line 182: Line 244:


func someFunc() SomeInterface {
func someFunc() SomeInterface {
   return &TheImplementation{} // this is a valid function declaration
   return &SomeImplementation{} // this is a valid function declaration
}
}
</syntaxhighlight>
</syntaxhighlight>


=Empty Interface=
=<span id='Empty_Interface'></span><span id='any'></span>Empty Interface <tt>interface{}</tt> <tt>any</tt>=
An empty interface is an interface that specifies no methods. Any type can "implement" that interface. If you want to declare a parameter of any type for a function, make it an empty interface:
An empty interface, declared as <code>interface{}</code> or <code>any</code> is an interface that specifies no methods. Any type can "implement" that interface. If you want to declare a parameter of any type for a function, make it an empty interface:
<syntaxhighlight lang='go'>
<syntaxhighlight lang='go'>
func SomeFunc(v interface{}) {
func SomeFunc(v interface{}) {
Line 193: Line 255:
}
}
</syntaxhighlight>
</syntaxhighlight>
<syntaxhighlight lang='go'>
func SomeFunc(v any) {
  ...
}
</syntaxhighlight>
According to the <code>builtin</code> package documentation, <code>any</code> is an alias for <code>interface{}</code> and it is equivalent to <code>interface{}</code> in all ways.


Rob Pike does not like empty interfaces: when you're programming and you use an empty interface, think really hard if this is actually what you want, or whether isn't just a little something you can put in, an interface with an actual method that is really necessary to capture the things you are talking about.
Rob Pike does not like empty interfaces: when you're programming and you use an empty interface, think really hard if this is actually what you want, or whether isn't just a little something you can put in, an interface with an actual method that is really necessary to capture the things you are talking about.
=<span id='Embedded_Type'></span><span id='Embedded_Types'></span>Interfaces as Embedded Types=
<font color=darkkhaki>Interface names can be used as [[Go_Structs#Embedded_Type|embedded types]] in structs.</font>
=Interface Inheritance=
{{Internal|Go_Language_Object_Oriented_Programming#Behavior_Inheritance|Behavior Inheritance}}


=Using Interfaces=
=Using Interfaces=
Line 204: Line 278:
}
}
</syntaxhighlight>
</syntaxhighlight>
=Type Assertions=
==Mocking==
{{External|https://go.dev/ref/spec#Type_assertions}}
See: {{Internal|Github.com/stretchr/testify#Mocks|Testify Mocks}}
 
An interface hides the differences between the types implementing it and emphasizes the commonality.
 
However, there are times you may want to know what exact concrete type exists behind the interface.
 
Type assertions can be used for type disambiguation:
 
<syntaxhighlight lang='go'>
var i SomeInterface
 
v, isType := i.(TypeName)
</syntaxhighlight>
 
Usage example:
<syntaxhighlight lang='go'>
type SomeInterface interface {
  SomeMethod()
}
 
type TypeA struct {
  s string
}
 
func (v TypeA) SomeMethod() {
  fmt.Println("TypeA.SomeMethod()")
}
 
type TypeB struct {
  s string
}
 
func (v TypeB) SomeMethod() {
  fmt.Println("TypeB.SomeMethod()")
}
 
func main() {
  var i SomeInterface
  i = TypeA{"A"}
  b := TypeB{"B"}
 
  a, isTypeA := i.(TypeA)
  fmt.Printf("%v, %t\n", a, isTypeA) // will print {A}, true
  b, isTypeB := i.(TypeB)
  fmt.Printf("%v, %t\n", b, isTypeB)  // will print {}, false
}
</syntaxhighlight>
 
Type assertion with <code>switch</code>:
<syntaxhighlight lang='go'>
var i SomeInterface
i = TypeA{"A"}
 
switch v := i.(type) {
  case TypeA:
fmt.Printf("TypeA: %v\n", v)
  case TypeB:
fmt.Printf("TypeB: %v\n", v)
}
</syntaxhighlight>
 
=TO DEPLETE=
 
<font color=darkkhaki>
==Internal==
 
* [[Go Concepts - The Type System#User-Defined_Types|The Type System]]
 
==Overview==
 
Go has an unique interface system that allows programs to model behavior rather than model types: the behavior is "attached" to types via methods, and, independently, the behavior contract is defined by interfaces. The fact that a type implements an interface is not formally declared via a keyword, like in Java, or otherwise, but is an implicit consequence of a type "having" all the behavior declared by the interface. This is called [[Programming_Languages_Concepts#Duck_Typing|duck typing]]. If a type implements an interface by the virtue of duck typing, a value of the type can be stored in a value of that interface type and the compiler ensure the assignments are correct at compile-time.
 
:''<span id="duck_typing_does_not_help_readability">Is this a good thing? As far as I can tell, I can't say whether a specific type implements an interface, short of browsing all methods in the package, looking for receivers of that specific type. The compiler does that easily, but I don't. This does not help code readability.</span>''
 
A type and its associated [[Go_Concepts_-_The_Type_System#Pointer_Types|pointer type]] may or may not implement the same interface, depending on how the methods are "associated" with the value and pointers of that type. In order to understand how a type or its corresponding pointer type implements an interface, you need to understand [[#Method_Sets|method sets]], which will be explained below. The notion of a type implementing an interface is related to method set inclusion. For more details see "[[Go_Interfaces#When_does_a_Type.2FPointer_Type_Implement_an_Interface.3F|When does a type/pointer type implement an interface?]]" below.
 
Any used-defined named type ([[Go Structs|structs]], [[Go Type Aliasing|aliased types]]) can be made to implement interfaces.
 
The fact that Go detaches the contract from implementation with interfaces allows for [[Programming_Languages_Concepts#Polymorphism|polymorphism]]. Wen a method call is made against an interface value, the equivalent method of the stored user-defined type value it is executed. The user-defined type that implements an interface is often called a ''concrete type'' for that interface.
 
Interfaces in Go tend to be small, exposing only a few functions, or just one function.
 
==Method Sets==
 
A type may have a ''method set'' associated with it.
 
There are two kinds of method sets: [[#Interface_Method_Set|interface method sets]] and [[#User-Defined_Type_Method_Set|user-defined types method sets]]. In the case of user-defined type method set, there's a distinction between the type method set and the pointer type method set, as it will be shown below.
 
If a type does not have any of the previously mentioned method sets, it is said to have an empty method set.
 
In a method set, each method must have a unique non-blank method name.
 
The language specification defines the method sets here: https://golang.org/ref/spec#Method_sets.
 
===Interface Method Set===
 
An ''interface method set'' is a list of method declarations (function name and signature), specified in the [[#Interface_Type|interface type declaration]]. The interface type declaration has the only purpose of defining the method set and giving it a name - the interface name.
 
The method set only contains method names and signatures, so the interface type does not define ''how'' the behavior is implemented, but just the behavior's contract. The interfaces allow us to hide the incidental details of the implementation. The behavior implementation details are contained by user-defined types, via methods.
 
Also, the method signatures from the interface method set are independent on whether the corresponding methods from the concrete types are declared with value or pointer receivers.
 
====Interface Type====
 
An ''interface type'' is a user-defined type whose only purpose is to introduce a method set (the [[#Interface_Method_Set|interface method set]]) and to give it a name.
 
The interface type declaration starts with the <tt>type</tt> keyword, to indicated that this is a user-defined type, followed by the interface name and the keyword <tt>interface</tt>, followed by the interface method set declaration between curly braces. The interface method set declaration consists in a list of method names followed by their [[Go_Concepts_-_Functions#Function_Signature|signatures]]. Unlike in the <tt>struct</tt>'s case, we don't define fields.
 
<pre>
type MyInterface interface {
    <function-name-1><function-signature-1>
    <function-name-2><function-signature-2>
    ...
}
</pre>
 
Example:
 
<pre>
type A interface {
 
    m1(i int) int
    m2(s string) string
 
}
</pre>
 
====Interface Name Convention====
 
If the interface type contains only one method, the name of the interface starts with name of the method and ends with the ''er'' suffix. When multiple methods are declared within an interface type, the name of the interface should relate to its general behavior.
 
====Interface====
 
The method set of an interface type it is said to be the type's ''interface''.
 
===User-Defined Type Method Set===
 
====Method Set associated with the Values of a Type====
 
The method set associated with the values of a type consists of all methods declared with a ''value receiver'' of that type.
 
The specification defines ''the method set of a type'' as the method set associated with the values of the type.
 
====Method Set associated with the Pointers of a Type====
 
The method set associated with the pointers of a type <tt>T</tt> consists of both all methods declared with a pointer receiver <tt>*T</tt> ''and'' with a value receiver <tt>T</tt>.
 
The method set associated with the pointers of a type always includes the method set associated with the values of the type. This is because given a pointer, we can always infer the value pointed by that address, so the methods associated with the value will naturally "work". The reverse is not true - given a value, not always we can get an address for it, and because we are not able to get an pointer, we cannot assume that the methods associated with the pointer will work. [https://github.com/NovaOrdis/playground/tree/master/go/reference This is an example that demonstrates that we cannot always get an address for a value].
 
Another way of expressing this is that pointer receiver method are ''less general'' than the value receiver methods: value receiver methods can ''always'' be used, pointer receiver methods can't always be used.
 
Also see:
 
<blockquote style="background-color: #f9f9f9; border: solid thin lightgrey;">
:[[Go_Concepts_-_Functions#Receivers_and_Interfaces|Receivers and Interfaces]]
</blockquote>
 
==When does a Type/Pointer Type Implement an Interface?==
 
Note that notion of a type implementing an interface and its corresponding pointer type implementing the same interface both make sense, and they are subtly different.
 
According to the language specification, a user-defined type <tt>T</tt> implements an interface if its [[#User-Defined_Type_Method_Set|method set]] (all methods declared with ''a value receiver'' of type <tt>T</tt>) is a superset of the [[#Interface|interface]].
 
By extrapolation, a pointer type implements an interface when the method set associated with pointers to that type is a superset of the [[#Interface|interface]]. As per [[Go_Interfaces#User-Defined_Type_Method_Set|User-Defined Type Method Set]] section, the method set associated with the pointer of a type include implicitly the method set associated with the value of the type.
 
For more details about value and pointer receivers, see:
 
<blockquote style="background-color: #f9f9f9; border: solid thin lightgrey;">
:[[Go_Concepts_-_Functions#Value_and_Pointer_Receivers|Value and Pointer Receivers]]
</blockquote>
 
For more details about pointer types see:
 
<blockquote style="background-color: #f9f9f9; border: solid thin lightgrey;">
:[[Go_Concepts_-_The_Type_System#Pointer_Types|Pointer Types]]
</blockquote>
 
==Example==
 
The example that follows shows how a type <tt>B</tt> implements interface <tt>A</tt>:
 
<pre>
//
// The interface A declares an interface method set containing a single method m()
//
type A interface {
    m()
}
 
//
// At this point, the type B is not yet linked to the interface A in any way.
// Its method set is empty, so it does not implement interface A, it does not
// implement any interface.
//
type B struct {
    i int
}
 
//
// We make B implement interface A by declaring B as a value receiver for m().
// This means B's method set includes now m() and it is a superset of A's method
// set.
//
func (b B) m() {
    // simply reports the value of i
    fmt.Println(b.i)
}
 
...
 
//
// B now implements A, so B instances can be assigned to A variables
//
var a A
a = B{1}
a.m()
 
//
// *B now also implements A, because the method set of the pointer type
// includes the methods declared agains the value  so *B instances
// can be assigned to A variables
//
var a2 A
a2 = &B{2}
a2.m()
 
...
</pre>
 
===A Variation of the Example===
 
Note that if in the example above we declare
 
<pre>
func (b *B) m() {
    ...
}
</pre>
 
instead of
 
<pre>
func (b B) m() {
    ...
}
</pre>
 
then the <tt>a = B{1}</tt> assignment would fail to compile with:
 
<pre>
./main.go:23: cannot use B literal (type B) as type A in assignment:
B does not implement A (m method has pointer receiver)
</pre>
 
This is because the type B does not implement the interface, because m() is not in the type B method set.
 
However, if we try:
<pre>
var a A
a = &B{1}
a.m()
</pre>
 
it works in both cases, because the method set associated with the B pointers include both <tt>func (b B) m() ...</tt> and <tt>func (b *B) m() ...</tt> so the pointer type B implements the interface.
 
==Passing Interfaces to Functions==
 
Interfaces can be used as arguments to functions. Passing an interface instance insures that the function body can rely on the fact the interface methods are available on the passed instance.
 
With the example above, we declare a function f() that expects an interface A and we pass a B instance when invoking the function:
 
<pre>
...
 
type A interface {
    m()
}
 
func (b B) m() {
    ...
}
 
func f(a A) {
    a.m()
}
 
...
b := B{1}
f(b)
...
</pre>
 
Note that the B instance can be passed by value (as in the example above) or by reference (as in the example below). Both cases work for reasons explained in the [[#Method_Set_associated_with_the_Pointers_of_a_Type|Method Set associated with the Pointers of a Type]] section:
 
<pre>
...
b := B{1}
f(&b)
...
</pre>
 
or (same thing)
 
<pre>
...
bPtr := new(B)
f(bPtr)
...
</pre>
 
==Interfaces as Fields==
 
Interfaces can be used as [[Go_Structs#Fields|fields]] in <tt>[[Go_Structs|struct]]</tt>s.
 
==Implementation Details==
 
An interface variable is a two-word data structure.
 
The first word contains a pointer to an internal table called ''iTable'', which contains type information about the stored value: the concrete type of the value that has been stored and a list of methods associated with the value. The second word is a reference to the stored value.
 
The combination of type information and pointer binds the relationship between the two values.

Latest revision as of 22:18, 5 November 2024

External

Internal

Overview

An interface is a set of method signatures. The interface is declared with the keyword type, followed by the name of the interface, followed by the keyword interface and by the method signatures enclosed in a curly brace block. Each signature contains the method name, optional parameter names and mandatory parameter types, and optional return variable names and mandatory return types. Providing parameter and return names is recommended, as it arguably exposes more semantics, making the code clearer. See example, below.

The interface declaration contains no method implementations.

Interfaces are not fully-featured types, they're less than that, but they're used to express conceptual similarities between types. If several types implement the same interface, they are similar in a way that is important to the application. An interface highlights the similarities between the types implementing it and conceals the differences.

An interface does need to be formally declared on a target type for that type to implement it. Interfaces are satisfied implicitly. A type implements or satisfies an interface if the type defines all methods specified in the interface. Of course, the type can have other methods not specified by the interface. This is called duck typing. If a type implements an interface by the virtue of duck typing, a value of the type can be stored in a value of that interface type and the compiler ensure the assignments are correct at compile-time. A type can implement multiple interfaces.

Interfaces can be thought of a kind of conceptual inheritance: types inherit behavior specified by an interface. Interfaces help supporting polymorphism in Go: different types implement different behaviors specified by the same interface. For more details see:

Go Inheritance and Polymorphism

Go interfaces allow programs to model behavior rather than model types. The behavior is "attached" to existing types, via methods, and independently, the behavior is given a name, the name of the interface that groups together the methods. Go type system is structural, not nominal. The interfaces are behavior contracts, and they belong in the consuming code, not in the producing code. A common pattern should be to accept interfaces and return structs.

This is debatable: for an interface itself, it feels natural that the methods should return the same interface, not an implementing type. What if the implementing type is package private, and the interface is public?

A type and its associated pointer type may or may not implement the same interface, depending on how the methods are associated with the value and pointer receivers of that type.

"The bigger the interface, the weaker the abstraction." The most important thing about interfaces in Go is that the smaller the interface is, the more useful it is. This is a Go-specific idea: we want to make little interfaces, so we can build components that share them.

Interfaces are defined at the "fault lines" of the system's architecture. These are between the main() function and the rest, and along the package API boundaries.

Interfaces should not be defined before a realistic example of usage exists.

Interfaces are rigid. For a library that exposes interfaces in the public API, any change to the interface requires releasing a new major version, since it breaks all third-party implementations.

Is that right? What if we add to the interface?

Declaration

The following syntax declares an interface:

type <InterfaceName> interface {
  // Interface method set:
  <method_signature_1>
  <method_signature_2>
  ...
}

Example:

type SomeInterface interface {
  MethodA(int) (int, error)                          // No parameter and return value names are specified.
  MethodB(someValue int) (someResult int, err error) // Parameter and return value names are provided.
                                                     // This is arguably clearer, as it conveys more semantics.
}

What results after such a declaration is an interface type. For a discussion of concrete type vs. interface types, see:

Go Language | Concrete vs. Interface Types

Interface Method Set

An interface method set the list of method declaration specified in the interface type declaration. The interface type declaration has the only purpose of defining the method set and giving it a name, which is the interface name. Also see:

Method Sets

Naming

Single-method interfaces are named using the method name plus an "er" suffix. An interface with just one Write method would be called Writer. Sometimes, the result isn't correct English, but we do it anyway. In other cases, we use English to make the result nicer:

type ByteReader interface {
    ReadByte() (c byte, err error)
}

When an interface includes multiple methods, choose a name that accurately describes its purpose.

Also see:

Go Style

Implementing an Interface

As mentioned above, interfaces are not formally declared to be satisfied with dedicated syntax, they are satisfied implicitly. The type satisfying an interface do not need to formally declare the name of the interface anywhere in its definition. Instead, it is sufficient to declare themselves as value or pointer receiver types for all the methods in the interface declaration.

type SomeImplementation { ... } // SomeInterface is specified nowhere in the SomeImplementation declaration

// Declaring the type as pointer receiver for *all* methods declared in the interface makes 
// the type to implement that interface . Value and pointer receivers must not be mixed by 
// the method implementations of an interface.
func (s *SomeImplementation) MethodA(i int) (int, error) { ... }
func (s *SomeImplementation) MethodB(someValue int) (int, error) { ... }

The method signatures from the interface method set are independent on whether the corresponding methods from the concrete types are declared with value or pointer receivers.

When Does a Type/Pointer Type Implement an Interface?

Note that notion of a type implementing an interface and its corresponding pointer type implementing the same interface both make sense, and they are subtly different. According to the language specification, a user-defined type T implements an interface if its method set, which are all methods declared with a value receiver of type T, is a superset of the interface. By extrapolation, a pointer type implements an interface when the method set associated with pointers to that type is a superset of the interface. As explained here, the method set associated with the pointer of a type includes implicitly the method set associated with the value of the type.

What Can Implement an Interface?

Structs can implement interfaces.

Type aliases can implement interfaces. In general, named types can implement interfaces, like in the following example where a bool can implement an interface as long as it is named:

type SomeInterface interface {
    SomeMethod()
}

type someImpl bool

func (s someImpl) SomeMethod() {
	fmt.Printf("SomeMethod\n")
}

someImpl(true).SomeMethod()

What else? Can a function be used as an interface?

Interface Values

Declaring a variable of an interface type creates an interface value:

var i SomeInterface // i is a pair: (dynamic_type, dynamic_value)

The interface value contains a data pair, including a dynamic type and a dynamic value. The dynamic type, when set, is a concrete type that implements that interface. The dynamic value, when set, is an instance of the dynamic type. Internally, an interface variable is a two-word data structure. The first word contains a pointer to an internal table called iTable, which contains type information about the stored value: the concrete type of the value that has been stored and a list of methods associated with the value. The second word is a reference to the stored value. The combination of type information and pointer binds the relationship between the two values.

In the case of the above declaration, both elements of the pair are nil:

fmt.Printf("%v\n", i) // prints <nil>

Assuming the following interface and concrete type declarations:

type SomeInterface interface {
  SomeMethod()
}

type SomeImplementation struct {
  data string
}

// SomeMethod makes SomeImplementation implement SomeInterface.
func (s *SomeImplementation) SomeMethod() {
  if s == nil {
    fmt.Println("invocation with a nil dynamic value")
  } else {
    fmt.Printf("data: %s\n", s.data)    
  }
}

... the following variable declaration and initialization ends up in an interface variable whose both dynamic type and dynamic value are set to *SomeImplementation and &SomeImplementation{"test"}, respectively:

var i SomeInterface
i = &SomeImplementation{"test"}

The interface value can be then used to invoke the SomeMethod() method on the dynamic value, which was assigned to the dynamic concrete type instance pointer.

i.SomeMethod() // prints "data: test"

An interface value can have a nil dynamic value, and still be usable, as long it is assigned a dynamic type with this syntax:

var i SomeInterface
var t *SomeImplementation
i = t

An interface value assigned with a pointer to the dynamic type, but not with a dynamic value can still be invoked into, given that the method implementation cases for the situation when the receiver type implicit argument is nil. In the example above, we assigned i a dynamic type *SomeImplementation but not a dynamic value, as t has no concrete value yet. The dynamic type is enough information to go find which implementation SomeMethod() should use.


This is why it is a good idea to guard against nil implicit argument in all concrete methods provided by types that implement interfaces.

This way we get a static method, in the Java or Python sense of the word and an instance method at the same time.

i.SomeMethod() // prints "invocation with a nil dynamic value"

An interface value that carries a dynamic type but no dynamic value is a legal state. It is a way to implement static (type) methods.

In the case of an interface variable with nil dynamic value, i == nil evaluates to false but testify require.Nil() evaluates to true.

This is how to test public interfaces whose implementing type is package-private:

Testing Public Interfaces whose Implementing Type is Package-Private

nil Interface Value

This describes an interface value with a nil dynamic type. In this situation, you cannot call the methods on that interface, because without a dynamic type you can't know which method you are referring to.

Type Assertion

There is syntax in the language which, given an interface variable, allows us to extract the dynamic value with the actual dynamic type:

Type Assertion

Type Switch

Type Switch

Interfaces as Function Parameters and Result Values

Interface as Function Parameter

Interfaces can be used as arguments to functions. Passing an interface instance insures that the function body can rely on the fact the interface methods are available on the passed instance. When a function is declared with an interface type parameter, the function can be invoked with either a value or a pointer of the implementing type, as long the implementation uses a value receiver type. For the following interface and type declaration:

type SomeInterface interface {
    MethodA()
}

type SomeImplementation struct {
    data string
}

the type may chose to implement the interface by using a value receiver:

func (s SomeImplementation) MethodA() {
    fmt.Printf("data: %s\n", s.data)
}

In this case, if a function is declared with a parameter of SomeInterface interface type, the function can be invoked by passing either a value or a pointer to the type instance:

func SomeFunc(i SomeInterface) {
    i.MethodA()
}

...

t := SomeImplementation{data: "blue"}
SomeFunc(t)

// this also works:
t2 := &SomeImplementation{data: "red"}
SomeFunc(t2)

If the implementation of the interface is done with a pointer receiver:

func (s *SomeImplementation) MethodA() {
    fmt.Printf("data: %s\n", s.data)
}

then only a pointer to the implementing type instance can be passed to the function SomeFunc():

t := &SomeImplementation{data: "green"}
SomeFunc(t)

If we attempt to pass an implementing type instance value, the compiler complains:

Cannot use t (variable of type SomeImplementation) as SomeInterface value in argument to SomeFunc: SomeImplementation does not implement SomeInterface (method MethodA has pointer receiver)

Interface as Function Result

Functions can return interfaces as results, as shown below. However, in general, functions should accept interfaces and return structs.

type SomeInterface interface {
  ...
}

type SomeImplementation struct {
  ... 
}

 // the type SomeImplementation  is presumably declared as receiver type for all interface methods, thus implements it 

func someFunc() SomeInterface {
  return &SomeImplementation{} // this is a valid function declaration
}

Empty Interface interface{} any

An empty interface, declared as interface{} or any is an interface that specifies no methods. Any type can "implement" that interface. If you want to declare a parameter of any type for a function, make it an empty interface:

func SomeFunc(v interface{}) {
  ...
}
func SomeFunc(v any) {
  ...
}

According to the builtin package documentation, any is an alias for interface{} and it is equivalent to interface{} in all ways.

Rob Pike does not like empty interfaces: when you're programming and you use an empty interface, think really hard if this is actually what you want, or whether isn't just a little something you can put in, an interface with an actual method that is really necessary to capture the things you are talking about.

Interfaces as Embedded Types

Interface names can be used as embedded types in structs.

Interface Inheritance

Behavior Inheritance

Using Interfaces

Function with Multiple Types of Parameters

Interface expresses similarity, so make the function take as a parameter an interface that expresses the similarity of all desired arguments.

func Area(s Shape) {
  ...
}

Mocking

See:

Testify Mocks