Go interfaces are a powerful feature that allows you to define a set of methods that a type must implement. This promotes flexibility and decouples code components, making your programs easier to maintain and extend. In this article, we will explore three practical examples of creating and using Go interfaces.
In this example, we will create a Shape
interface that can be implemented by various geometric figures like circles and rectangles. This demonstrates how interfaces can help in managing different types that share common functionality.
package main
import (
"fmt"
"math"
)
// Shape interface declares methods for calculating area and perimeter.
type Shape interface {
Area() float64
Perimeter() float64
}
// Circle struct implements the Shape interface.
type Circle struct {
Radius float64
}
// Area calculates the area of the circle.
func (c Circle) Area() float64 {
return math.Pi * c.Radius * c.Radius
}
// Perimeter calculates the perimeter of the circle.
func (c Circle) Perimeter() float64 {
return 2 * math.Pi * c.Radius
}
// Rectangle struct implements the Shape interface.
type Rectangle struct {
Width float64
Height float64
}
// Area calculates the area of the rectangle.
func (r Rectangle) Area() float64 {
return r.Width * r.Height
}
// Perimeter calculates the perimeter of the rectangle.
func (r Rectangle) Perimeter() float64 {
return 2 * (r.Width + r.Height)
}
func main() {
shapes := []Shape{
Circle{Radius: 5},
Rectangle{Width: 3, Height: 4},
}
for _, shape := range shapes {
fmt.Printf("Area: %.2f, Perimeter: %.2f\n", shape.Area(), shape.Perimeter())
}
}
Circle
and Rectangle
can be treated as Shape
types.In this example, we will create a Logger
interface that allows us to implement multiple logging strategies, such as console logging and file logging. This is useful for applications that require different logging mechanisms.
package main
import (
"fmt"
"os"
)
// Logger interface declares a method for logging messages.
type Logger interface {
Log(message string)
}
// ConsoleLogger implements the Logger interface for console output.
type ConsoleLogger struct {}
// Log prints the message to the console.
func (c ConsoleLogger) Log(message string) {
fmt.Println(message)
}
// FileLogger implements the Logger interface for file output.
type FileLogger struct {
file *os.File
}
// Log writes the message to a file.
func (f FileLogger) Log(message string) {
f.file.WriteString(message + "\n")
}
func main() {
consoleLogger := ConsoleLogger{}
file, _ := os.Create("log.txt")
fileLogger := FileLogger{file: file}
loggers := []Logger{consoleLogger, fileLogger}
for _, logger := range loggers {
logger.Log("This is a log message.")
}
defer file.Close()
}
Logger
interface allows us to switch logging strategies easily.In this example, we will create a PaymentProcessor
interface that can be implemented by various payment methods such as credit cards, PayPal, or cryptocurrency. This showcases how interfaces can facilitate different implementations in an e-commerce setting.
package main
import (
"fmt"
)
// PaymentProcessor interface declares a method for processing payments.
type PaymentProcessor interface {
ProcessPayment(amount float64) string
}
// CreditCardProcessor implements the PaymentProcessor interface for credit card payments.
type CreditCardProcessor struct {}
// ProcessPayment handles payment processing for credit cards.
func (cc CreditCardProcessor) ProcessPayment(amount float64) string {
return fmt.Sprintf("Processed credit card payment of $%.2f", amount)
}
// PayPalProcessor implements the PaymentProcessor interface for PayPal payments.
type PayPalProcessor struct {}
// ProcessPayment handles payment processing for PayPal.
func (pp PayPalProcessor) ProcessPayment(amount float64) string {
return fmt.Sprintf("Processed PayPal payment of $%.2f", amount)
}
func main() {
processors := []PaymentProcessor{
CreditCardProcessor{},
PayPalProcessor{},
}
for _, processor := range processors {
fmt.Println(processor.ProcessPayment(100.00))
}
}