Large Blog image

Dependency Injection and Autowiring in Go: A Necessary Evil or a Feature We Don’t Need?

How I Ended Up Creating a Dependency Injector, the Journey from PHP to Golang


Listen to the Blog and Follow Along with the Transcript


Transitioning from PHP to Golang has been an eye-opening experience. In PHP, dependency injection (DI) is widely used, especially with frameworks like Symfony and Laravel. However, when I started working with Golang, I realized that while DI exists, it is not as commonly used. This made me wonder: Why is dependency injection so popular in PHP but not in Golang?

Why Dependency Injection is Popular in PHP

PHP is a dynamically typed, interpreted language that is heavily object-oriented. Frameworks like Symfony and Laravel follow the SOLID principles, encouraging the use of dependency injection to achieve:

  • Better maintainability – Injecting dependencies makes it easy to swap implementations without modifying the dependent classes.
  • Testability – Mocking dependencies is simpler, making unit testing easier.
  • Decoupling – Reduces tight coupling between classes, leading to a more modular design.
  • PSR Standards – The PHP-FIG group introduced PSR-11 (Container Interface), which provides a standard way for dependency containers to work.

Why Golang Doesn't Emphasize Dependency Injection

Golang takes a different approach:

  • Minimalist and explicit philosophy – Golang prefers explicit over implicit, avoiding magic.
  • No built-in OOP system – While Golang has interfaces and structs, it does not have class-based inheritance like PHP.
  • Dependency injection is done manually – Since Golang uses composition over inheritance, dependencies are often passed directly in constructors.
  • Static typing with compilation – Reflection is more limited and less performant compared to dynamically typed languages.

Some references discussing DI in Golang vs PHP:

What is Autowiring?

Autowiring is a feature commonly found in PHP DI containers where dependencies are automatically resolved based on type hints. It removes the need for manual configuration.

Pros of Autowiring:

  • Reduces boilerplate code.
  • Makes dependency management easier.
  • Works well with reflection-based frameworks.

Cons of Autowiring:

  • Can introduce hidden dependencies, making debugging harder.
  • Performance overhead due to runtime reflection.
  • Less explicit, which can be problematic in large projects.

The Motivation Behind My Dependency Injector

While exploring Golang DI solutions, I wanted something that felt familiar to my PHP background. However, I couldn't find an implementation that provided the same experience. So, I thought: Why not build my own? 🚀

I aimed to:

  • Introduce constructor-based dependency resolution.
  • Support autowiring to minimize manual dependency configuration.
  • Follow PSR-like standards where possible.

This was more challenging than PHP due to Golang’s:

  • Lack of runtime type information.
  • Limited reflection capabilities.
  • Compiler enforcing strict type checks.

But hey, I love a good challenge! 💪

How It Works

Below is an overview of my dependency injection container, godi.

Dependency Injection in Golang

Creating the Dependency Container

container := godi.New()

Constructor-based Dependency Resolution

func (t *yourStruct) Construct(param yourInterface) {}

Registering Dependencies

container.Set("TestInterface", NewTest2())

Resolving Dependencies

test := NewTest()
_, err := container.Get(test)
if err != nil {
 fmt.Println(err)
}
test.DoWhatYouWant()

Autowiring

type exampleMultipleImpl struct {
 Dependency1 exampleMultipleDepInterface `di:"autowire"`
 Dependency2 exampleMultipleDepInterface `di:"autowire"`
 Dependency3 exampleMultipleDepInterface `di:"autowire"`
}

Managing Dependencies

Adding a Single Dependency

container := godi.New()
container.Set("TestInterface", NewTest2())

Retrieving a Single Dependency

dependency, err := container.GetDependency("dependencyInterfaceName2")

Adding Multiple Dependencies

dependencies := map[string]interface{}{
 "dependencyInterfaceName1": NewDependency1(),
 "dependencyInterfaceName2": NewDependency2(),
 "dependencyInterfaceName3": NewDependency3(),
}
container.Build(dependencies)

Calling Functions with Resolved Dependencies

result, err := container.Call(TestFunc)
func TestFunc(d1 dependencyInterfaceName1, d2 dependencyInterfaceName2, d3 dependencyInterfaceName3) {}

Custom Values in Function Calls

result, err := container.Call(TestFunc, 55, "Hello")
func TestFunc(intValue int, stringValue string, d1 dependencyInterfaceName1, d2 dependencyInterfaceName2, d3 dependencyInterfaceName3) {}

Non-Singleton Resolution

container.Set("exampleInterface", callerFunc)
func callerFunc() interface{} {
 return Your()
}

Flushing Dependencies

container.Flush()

Deleting a Single Dependency

container.Delete("dependencyInterfaceName2")

Counting Dependencies

count := container.Count()
fmt.Println(count)

Is This Useful? Let’s Discuss!

So, that’s my journey in creating a Golang DI container with autowiring. It was a fun challenge, and I’d love to hear your thoughts:

  • Do you think DI is useful in Golang?
  • Would you use autowiring, or do you prefer manual dependency injection?
  • What other features would you like to see?

Let’s start the discussion in the comments! 🚀