Need of Dependency Injection
Traditional Approach of Creating Application:
Issue with above Approach:
- 1. It is difficult to maintain over time (tightly Coupled)Over the time if Data Access class change its constructor at that time it break the Business Layer classes which are dependent on that class & it is hard to manage in every changes.
- 2. Hard to Unit TestIt is hard to mock Data Access layer classes in unit test case of Business layer classes & make it complex.
Solution:
What is Dependency Injection?
data:image/s3,"s3://crabby-images/5f3ce/5f3ce944adf90e0333fffbc7921c90e857efffdb" alt=""
data:image/s3,"s3://crabby-images/896ed/896ed30fbb194f0717de1ece4866c11577747b98" alt=""
In above example we can see that in tightly coupled class are there. And if any changes happen in Item class Constructor then we need to change in Order class as well. And in dependency injection you no need to worry about any changes happen in Item Constructor it responsibility of Dependency Injection Container which create object of item class at time of order object create and so on it also take care if Item class also dependent on another class.
There are three types of dependencies:
- Constructor
- Setter (Property)
- Method
Note: In this blog I am showing Constructor Injection with Unity Framework (that is mostly widely used).
Dependency Injection Container
A Dependency Injection Container is an object that knows how to instantiate and configure objects. And to be able to do its job, it needs to know about the constructor arguments and the relationships between the objects. It is responsible for create new object and return whenever it required or requested.
Implementation of Dependency Injection with example
I am going to create one console application by following steps.
1. Create Console Application in visual studio by creating New Project with name “DIProject”.
data:image/s3,"s3://crabby-images/7e442/7e4429d2b5b0f98cb3d045adef446530cdce585d" alt=""
2. Adding the Reference of Microsoft Unity Framework using Nuget Package Manager.
data:image/s3,"s3://crabby-images/e27e2/e27e213279c8b22b5e21acdd78d99ec1d43a1474" alt=""
3. Now I am adding Order Class which is responsible for putting Order of Item.
data:image/s3,"s3://crabby-images/516f5/516f54e1037fa97e3d145bd774e41acfccde774b" alt=""
4. Create IProducts Interface which is implemented by item classes.
data:image/s3,"s3://crabby-images/a429f/a429fc54943920c8c021106adc7392174adfc0d3" alt=""
5. Same as add 2 Item Class Tea and Coffee both Implemented IProducts interface.
data:image/s3,"s3://crabby-images/aaf2a/aaf2a316f41bc20fa787b8264ed92be78edeb41e" alt=""
data:image/s3,"s3://crabby-images/96b65/96b6533c8e99a989864ec93284a93a9eb38284be" alt=""
6. Now In Order Constructor we inject IProduct interface and create putTeaOrder method.
data:image/s3,"s3://crabby-images/f129e/f129e3490f949fd10c33dcc7ab8be8e6875d95fa" alt=""
7. Now we need to configure Unity Container so we can it provide object at time of resolve Interface.
data:image/s3,"s3://crabby-images/3d533/3d53301e9a87f9e22bbbfc254194b68af7e753c7" alt=""
8. Now resolve dependency from container by using Resolve method of container as mention below for Order.
data:image/s3,"s3://crabby-images/571a8/571a878b14ccf9219a5170c2ac06d157e7063bf8" alt=""
9. Now Calling PutTeaOrder Method of Order class.
data:image/s3,"s3://crabby-images/a1a25/a1a25610c0ff4480ddb5e8b669e984b42b0036d4" alt=""
10. Snapshot of entire configuration & calling method.
data:image/s3,"s3://crabby-images/9303c/9303c59b9507dfefd0eea62f7f053d5b1cb13de5" alt=""
data:image/s3,"s3://crabby-images/f4c50/f4c50ef83cd32fb3aa877e72c621b4f4440ce488" alt=""
data:image/s3,"s3://crabby-images/38ab7/38ab7e2ad3910e15f12c43fa487ff9a38c424067" alt=""
Now you have clear idea about Dependency Injection & how it configurable, And Order Class behave as independent class if any one change constructor of Tea class then also no need to change Order Class as all thing manage by Unity Container.
Register Single type with multiple Classes & Runtime Resolve Object
Continue with above example, have you notice that I implement IProducts Interface in two classes Tea and Coffee but I use only one Tea. Now we change requirement that we need to add one more product Coffee required to add in order. For this we already created class Coffee.
1. Register Type single type with two classes as mention below, it also call Name Binding.
data:image/s3,"s3://crabby-images/09848/09848a87093874ddc166dd49e1707619add9efe5" alt=""
data:image/s3,"s3://crabby-images/70ce3/70ce3b997f376d0b3b109aefac0b9cb4fb84419d" alt=""
data:image/s3,"s3://crabby-images/8361a/8361a45f19381f4f16c57884a3aa979fec0c2c8a" alt=""
data:image/s3,"s3://crabby-images/a1852/a185210b10a9655a01e3c0fc939c7170a8064f88" alt=""
data:image/s3,"s3://crabby-images/80d44/80d44ef84c583f8cc699fb2aba03d8a7b7b7390e" alt=""
data:image/s3,"s3://crabby-images/bc80b/bc80b9081b6f7ac0d265dadfb873ffbf1aa22ff5" alt=""
Dependency Injection Pros and Cons
Pros | Cons |
---|---|
Loosely Coupling |
Increase code complexity |
Increase Testability |
Complicate debugging of code |