It's always good to revisit topics that you already touched before, the chance is that you will find out new things or remember forgotten ones.
So while I go over The Well-Grounded Java Developer, I intend posting here my own examples on different topics and techniques covered there.
Today, the theme is going to be Dependency Injection. You probably know that there's a specification for this, namely JSR-330 and you surely know what Dependency Injection is (let's call it DI from now on).
If not, to put it short, DI allows you make your classes and their dependencies loosely coupled. Let's say you have the class Company which has one Employee:
In this case, the Company is forced to use all the time the same type of Employee. You cannot easily 'hire' in place an ExperiencedDeveloper, subclass of Employee, without changing the whole Company. So what if we could inject any kind of Employee into the Company, without affecting the logic of the Company?
The simplest DI example is the constructor which gets an instance of the dependency class:
But wouldn't be nice to have something to take care for you of instantiating your dependencies? That's where Inversion of Control containers come in, the favorite of the day: Google Guice
By the way, Google Guice is the Reference Implementation for the JSR-330 Dependency Injection in Java
specification so if you want something more advanced you might want to take a look at Spring and PicoContainer.
These being said, let's go to our example. Let's say we have a Farmer, who has a Farm and a Land. Every year he works the land and he gets _Crop_s. But the _Crop_s he gets depend on the type of Weather he has and the Weather is controlled by God. So we can safely say (figuratively speaking), that God injects the Weather in his Farm. And the Farmer would have to manage with whatever he gets, right?
Let's put this into code! The complete project can be found here on my GitHub account.
So we have a Farm:
Notice the @Inject
annotation before the constructor, that will inject our Weather into our Farm. Here's the Weather interface which has 3 implementations(DryWeather, MildWeather and WetWeather):
You probably guessed, different weather implementation, different types of _Crop_s in different quantities. But who says what Weather should be injected and from were? We would need a com.google.inject.AbstractModule
implementation for that. In our case, the Land:
The Land says that a com.google.inject.Provider
, in our case GodFactor, is responsible for providing our needed Weather instances:
I'm not sure God makes his decisions this way, but for this case let's say that the type of Weather is decided considering a random number and it's rest after the division by 3.
The last important piece in our little experiment is the Farmer. How does he manage his Farm?
The first thing he must do is get the Injector
from Guice and use it to get an instance to its Farm and then collect the _Crop_s. So if we run this nice Farm for 5 years (the rest of the classes I didn't present, Weather subclasses and Crop, you can see on the GitHub repo) we would get something like this:
But wait, our poor Farmer gets a different Farm each year and also the Weather is totally different! That's not normal! Fortunately, we can do something about that. We can set the Scope of the DI to be Singleton, the only implementation of the Scope class. (Web apps get SESSION and REQUEST scopes as well, but those are in a special jar from Guice with web components.)
In order to do this, we modify the Land class:
Run it now and you get an output similar to this:
And that would be the end of my little farming with Guice experiment. I hope it helps to easily understand which are the components needed to use DI and which is their role. This example could have been done in a number of other ways, but I let you to discover which are those. :)
You can read much more about DI online, there are plenty of resources, a Google search away, or to get started you can check out the The Well-Grounded Java Developer book.