Inversion of Control Principle using Kotlin
Inversion of Control (IoC) is a software design principle that helps reverse control in object-oriented programming. Control here is any extra responsibility a class has besides its primary responsibility. <!--more-->
Introduction
IoC controls an object's creation and the flow of an application to a container or framework. IoC inverts the control of creating and managing objects from the programmer to a container. This container manages the creation and life-cycle of objects in the application. IoC helps us create large systems by eradicating the responsibility of creating objects.
Imagine this, suppose you have a bakery that specializes in making egg rolls. What is an egg roll without an egg? So eggs are essential in the making of egg rolls. Why egg rolls, you may ask? Well, you could call it plain nepotism.
So how do we get a bakery that obeys the IoC principle? Well, rather than having a poultry farm and involving yourself with all the poultry-related tasks, you order the eggs and get them delivered to you. This time, you don't have to worry about the chicks and the eggs being healthy or using suitable animal feed. With this new design, you can focus on making egg rolls, which results in a more effective baking process.
This is the big idea behind IoC: separating concerns. Inversion of control makes it possible for a consumer to have more control over the software. It gives it more flexibility and freedom to choose from other options.
Prerequisites
Before we begin, you should have:
- Basic knowledge of Kotlin programming language.
- An IDE installed. Preferably IntelliJ.
Some benefits of using IoC
- It is easy to switch between different implementations of a particular class at runtime.
- It increases the modularity of the program.
- It manages an object’s life-cycle and configuration. Some objects can be singletons, while we can create others per request.
Before we look at some code, here are some things you should have in mind:
1. Coupling
Coupling is a measure of how closely bound software modules are to one another. It is the degree to which changes made to a component force a need for a change in other components or modules.
Tight coupling is when a change in component A requires a change in component B. Loose coupling means that components A and B are independent. Thus a change in component A would not affect B. Loose coupling is often a sign of testable, well-structured, maintainable, and readable software. Tight coupling leads to code fragility and code rigidity.
Code fragility is the likelihood that software will break in many places every time there is a change.
Code rigidity is the degree of difficulty in making changes to the software.
Advantages of Loose Coupling
- Developers can quickly evolve loose coupled software because it encourages many changes without breaking the existing code.
- Loose coupling enhances software agility because it promotes iteration, meaning one can quickly add a new feature or functionality.
- Loose coupling reduces technical debt.
2. Abstraction
When you call something abstract, it means that it is incomplete or not clearly defined. Abstraction is a programming approach that hides implementation details, revealing only the functionality (relevant operations) to the user. It is one of the basic concepts in object-oriented programming.
Advantages
- It simplifies programming complexity.
- It promotes the grouping of related classes and objects.
3. Single Responsibility Principle
This principle states that every function, class, or module should have a single reason to change and have only one responsibility.
Advantages
- It makes the code easy to understand, fix, and maintain.
- Classes are less coupled and more resilient to change.
- More testable design.
4. Dependency Inversion Principle
The Dependency Inversion Principle (DIP) enables us to create loosely coupled systems. Making them easy to change and maintain.
DIP states that:
- High-level modules shouldn't depend on low-level modules. They should instead rely on abstractions.
- Abstractions shouldn't depend on details, but details should depend on abstractions.
High-Level modules are modules written to solve problems and use cases. They are abstract and a map to the business domain(business logic). Their concern is with what the software should do and not how they should do it.
Low-Level Modules are implementation details required to execute the business policies (logic). They are the plumbing or internals of a system, and they tell us how the system(software) should do various tasks. They tend to be very concrete.
For Class A to work, it depends on two low-level modules: Class B and Class C. This doesn't follow the dependency inversion principle because Class A, which is a high-level module, depends on Class B and C, which are low-level modules.
To make this code obey the DIP principle, extract the interface for the low-level modules. Extracting the interface will give us something that looks like this:
This abstraction could be an interface or an abstract class. Now we can head back to the main focus—Inversion of Control.
Inversion of control only provides design guidelines and not implementation details. A design principle is not constricted to any programming language. You can implement it whichever way you please. However, design patterns recommend an actual implementation. Design patterns are more like reusable solutions to a problem in a given scenario.
Thus, we can apply IoC in object-oriented programming in many ways. Some of which are:
- Dependency injection pattern.
- Strategy design pattern.
- The service locator pattern, and many others.
Let's look into dependency injection and strategy design patterns.
1. Dependency Injection (DI)
Dependency Injection(DI) is a design pattern primarily used with the Dependency Inversion Principle. It makes it possible for dependent objects to be created outside of a class. It then provides those objects to the class.
For example, we have a class LoginManager
that depends on the implementation of UserRepository
.
class LoginManager {
val userRepository: UserRepository = UserRepositoryImpl()
}
We can see that there is a high dependency between LoginManager
and UserRepository.
LoginManager
is directly dependent on the UserRepository
because UserRepository
handles its creation. This violates the Dependency Inversion and Single Responsibility Principle. The result is a tight couple between LoginManager
and UserRepository.
There are a couple of ways to fix this:
- Using public setters: This is not recommended because it might leave objects in an uninitialized state.
class LoginManager {
lateinit var userRepository: UserRepository
}
fun main(args: Array) {
val loginManager = LoginManager()
loginManager.userRepository = UserRepositoryImpl()
}
- Declare all the dependencies in the component's constructor: This would look like this:
class LoginManager (val userRepository: UserRepository){
//Do something...
}
fun main(args: Array) {
// In the caller function, create an instance of UserRepository
val userRepository = UserRepositoryImpl()
// use the UserRepository instance to construct an instance of LoginManager
val loginManager = LoginManager(userRepository)
}
LoginManager
now has a constructor that accepts UserRepository
abstraction as a parameter. LoginManager
can now use UserRepository
whichever way it pleases because UserRepository
is now a field in the LoginManager
class.
LoginManager
is no longer responsible for creating its dependencies. It is now the caller's job in this case—the main function. The main function provides the required dependency, which is UserRepositoryImpl.
This way, we can have different implementations of LoginManager
and quickly test it in other contexts.
Let's look at a more complex scenario:
We can see that Class A and B have no dependencies. Class C is dependent on Class A. Class D is dependent on Class B, and Class E is dependent on both Class C and D. If we want to call a method on or create an instance of class E, we would have to create all its required dependencies. We need to create concrete instances of Classes A, B, C, and D in a particular order.
Firstly, we would create instances of Class A and Class B because they have no dependencies. Next, we would create instances of classes C and D because we have instances of their respective dependencies A and B. Finally, we can create an instance of Class E.
Whew! Right? This is a simple example with just five classes. Imagine what would happen with real-life projects. There could be hundreds of classes that have to be instantiated every single time. That would be a whopping load of redundant work. Dependency injection is a great technique needed for achieving loosely coupled classes. But we can see here that manually doing dependency injection is not such a good idea.
Also, if you want to consider the lifecycle of these objects, supposing you want Class C to be a singleton and create D on every request. Handling this request manually would involve a lot of logic and redundant codes. This is where manual dependency injection comes in. With manual DI, you can create your own dependencies container class. This container will house your application's dependencies.
The IoC or DI Container now controls the creation and lifecycle of the objects. This way, you don't have to create instances of these objects every single time you need them anymore. Dagger and hilt can automate this process and generate the necessary code for you.
IoC Containers are primarily used in an application for objects like:
- Services.
- Data Access.
- Controllers.
It would be best not to create instances of entities, data transfer, or value objects in containers. You can always create new instances of them when needed, which is okay from an architectural point of view.
2. Strategy design pattern
A strategy pattern is a behavioral design pattern. It makes it possible to change the behavior of a class or its algorithm at run time.
Assuming we have an interface called LaundryBot
, it will have methods for washing, drying, and folding. Different fabric types will use this interface.
An example is the CashmereLaundryBot
and SilkLaundryBot
used for laundering cashmere and silk.
We should have in mind that:
- All items supplied to the laundry can either be hand-washed or machine-washed.
- All items supplied to the laundry can either be sun-dried or dried using a dryer.
- All items are being folded in the same way, which means that fold will be a default method in the
LaundryBot
interface.
interface LaundryBot{
fun wash()
fun dry()
fun fold(){
//fold items
}
}
We said an item could either be hand or machine washed and either sun-dried or dried using a dryer (machine dried), meaning that any LaundryBot
class would have either of these methods hence if we have a DenimLaundryBot
, and a CottonLaundryBot
they would both have the same piece of code for their wash
and dry
implementations of LaundryBot
. This is because both denim and cotton will be machine washed and machine dried.
How do we take care of this code duplication?
Now, this is where the strategy pattern comes in. The Strategy pattern encapsulates a set of algorithms that are interchangeable at runtime. How do we implement this in LaundryBot
? To do this, we would remove the Wash()
and dry()
methods from the LaundryBot
interface. We would then make them interfaces called Wash and Dry. These interfaces would have methods Wash()
and dry()
respectively.
interface Wash{
fun wash()
}
interface Dry{
fun dry()
}
Next, we provide concrete implementations of these interfaces. This way, we can encapsulate the different behaviors for each case of wash and dry. To do this, create classes called MachineWash
and HandWash
that inherit from wash
. In the same way, MachineDry
and SunDry
will inherit from Dry.
object MachineWash: Wash{
override fun wash() {
//Perform Machine Wash
}
}
object HandWash: Wash{
override fun wash() {
//Perform Hand Wash
}
}
object SunDry: Dry{
override fun dry() {
//Perform Sun Dry
}
}
object MachineDry: Dry{
override fun dry() {
//Perform Machine Dry
}
}
We have successfully created a family of algorithms for wash and dry.
We have created strategies; now, it's time for us to see how it works. Remember, we removed the Wash()
and dry()
methods from the LaundryBot
interface. This time, rather than having Wash()
and dry()
, we would create fields of type Wash and Dry.
interface LaundryBot {
val wash: Wash
val dry: Dry
fun fold() {
//fold items
}
}
This way, any class inherits from LaundryBot
can choose the particular behavior it wants from the algorithm family. Let's see how this works. Say we want to create a CashmereLaundryBot
. What behaviors of Wash and Dry do we choose? It is advised to wash cashmere by hand to avoid tears and snags; thus, we would use the HandWash behavior.
For drying, cashmere isn't suitable for MachineDry because the heat could shrink or damage its follicles. The Dry behavior to use for a cashmere then is the SunDry. Let's see how this looks like in code:
class CashmereLaundryBot: LaundryBot{
override val wash: Wash = HandWash
override val dry: Dry = SunDry
}
See how the strategy pattern has made it easy to change the behavior of this class? Nice!
Conclusion
Inversion of control is a practical approach to improving code modularity, reducing code duplication, and simplifying testing. Although it is beneficial when developing reusable libraries, it is not appropriate in all use cases.
It is essential to know when to take advantage of the flexibility and freedom inversion of control and when not to. It's been a long article, but I sure hope this will help you as much as it helped me.
Happy learning!
Peer Review Contributions by: Briana Nzivu