Dependency Injection Part 0 - Sat, Aug 28, 2021
Whats the Deal?
Part 0: What’s the deal? <– (You are here!)
So, you’ve heard your coworkers, peers, or anonymous internet friends mention Dependency Injection. Sometimes off the cuff as part of a sentence filled with other magic tech mumbo jumbo.
It sound’s pretty cool though, right? ’Dependency Injection', you’re veritable key to the programmer’s castle. It must be important to your code, right?
Well, not really. In fact, for a large quantity of code architecures, especially Functional Programming, Dependency Injection is out of the question since it goes against the very principle of your language.
But if you’re using an Object Oriented Architecture like C# or Java, well, now we are talking.
Let’s start by seeing what the internet has to say about DI (what we will refer to Dependency Injection by from now on because, well, we programmers love our Achronyms.
“In software engineering, dependency injection is a technique whereby one object (or static method) supplies the dependencies of another object. A dependency is an object that can be used (a service). An injection is the passing of a dependency to a dependent object (a client) that would use it.” - Wikipedia
Well alright thats pretty straightforward. Except probably not, right? So what does all of that mean?
Well, to start, hopefully if you’re here reading this you’re at least familiar with things like classes, constructors, ‘new’ing up something, instantiation, etc.
If not, go learn all those things first and come back please.
So I’ll assume you’ve got the basics and you’re ready for taking on some real programming architectures.
To start, consider you’re making a videogame, and you have the following Character
class representing a character, and a static class called AttackService
you use for, stuff or whatever:
|
|
Well, an experienced programmer will quickly alert you to the fact that simply using Static classes with Static methods is cumborsome, hard to maintain, requires loading things up front even if you don’t need it, hard to unit test properly… It’s a headache. Ok so you move to having AttackService
as a non static class, your Character
class probably looks like this now…
|
|
Better, right? Well sure maybe, if you’re okay with the fact that if someone accidently breaks AttackService
.CalculateDamage
, any unit tests for methods that call AttackService.CalculateDamage
will also false flag as broken too, making it hard for you to zero in on where the real problem was.
Ok, so how do you get around this? How could you possibly have a class use another class, without breaking when that class breaks? Enter dependency injection (and hand in hand with it, Mocking Unit Test architecture!)
How To Inject
So step one is, chances are your Object Oriented language of choice has something along the line of interfaces, which are effectively contracts you sign your classes up for that say “My class will have these publically exposed things at minimum”
So lets start by making the IAttackService
interface, which has that public method on it, and bind it to our AttackService
implimentation.
|
|
Ok cool, that doesn’t look like it changed anything, right? Wrong! See the thing about IAttackService
, is it has no idea what CalculateDamage
actually does, nor does it care what CalculateDamage
does. Which might start giving you the hint about what we are going to do next… One little tweak to our Character
class and…
|
|
Now you’ll probably notice, when you need to ‘new’ up a Character you’ll have to do it like this now:
|
|
And that, my dear reader, is Dependency Injection
. Character class is Dependant on IAttackService
, but you have Injected
it through the Constructor
This type of Dependency Injection is called Constructor
Dependency Injection.
If you wanted the Dependency to be optional, perhaps with a default, you also could do Property Injection like so:
|
|
Which also works fine. But we’ll just stick with Constructor Injection for now because, in my opinion, it’s cleaner and easier to maintain.
Congratulations, you now understand the basic of DI!
#But why?
Ok fair question, so consider now you want to unit test Character
class without fear that your unit tests will fail. Well, rather than just injecting the usual AttackService
, what if you constructed a dummy ‘fake’ one that just did everything in a simple, predictable way? And that fake service only existed inside of your unit test scope, so no one outside of it could accidently use it?
Welcome dear reader, to the wonderful world of Mocking! You would do it in our example as simple as this:
|
|
Boom, easy as that! You could even do something fancy like, perhaps add the following to your MockAttackService…
|
|
And take your Unit test to the next level by validating your Dependency usage!
|
|
Muh News!
Ok so this all seems pretty awesome so far, but now let me show you the big ugly truth to dependency injection, and how modern programming has solved it. Imagine you have a rather large program with many many lines of code, hundreds of classes, and lots of nested dependencies.
Suddenly, simply naively ‘new’ing up a new class with deep nested dependencies could look as nasty as this:
|
|
Oh geez! This is really starting to get out of hand now. But you probably are noticing that, hey, all of this dependency injection just looks the same all the way down. What if there was an easy way to just automatically construct all these things? Like some kind of magical Service Container
that had everything we could ever use setup in it, and when we asked for one it automatically injected everything for us all the way down the chain…
Maybe our magical code then would just become:
|
|
Well, this exists, in many different versions. Autofac
, Ninject
, ServiceCollection
, the list goes on and on.
This is called a Dependency Injection Container
, and you generally set it up by first registering all the classes it should know about at the start of your program, and once those are ‘logged’ into it, you can just summon them back out at will later.
Pretty neat, huh?
Lets make one.
Which will lead us on to Part 1!