Featured Articles
Article in Android App Development, Mobile App Development categories.
Dependency Injection: Dagger and Koin
Dependency Injection on Android has been a hot button topic. Opinions on the topic range from swearing by Dagger to using a host of new…
Dependency Injection on Android has been a hot button topic for quite a while now. Opinions range from not using DI at all, to using manual DI, to swearing by Dagger, to using the host of new libraries that have been popping up.
I explored Koin recently and wanted to share some thoughts regarding the experience. It also led me to have a better understanding of Dagger and DI in general. So, let’s get down to it! First, I will lay down a basic comparison of how we perform DI with both the libraries.
Disclaimer: This article assumes you are familiar with the concept of DI and Dagger. If not, feel free to go through this post. It also links to a video series which helped me immensely in understanding Dagger.
Modules
Both the libraries treat modules a bit differently. In the case of Dagger, you’ll need modules to provide objects whose constructor cannot be annotated with @Inject
. But for Koin, all dependencies must be declared in a module.
So looking at Koin first, here’s how you would define a module:
On the other hand, here’s what we got for Dagger:
As you can see right away, the code we have to write is much more verbose for Dagger. And I am not even including the code for providing ViewModels 😅
Components
Next, let’s take a look at components.
Koin doesn’t have Components per se. You just have this:
The startKoin
method which takes in a list of your modules (and optionally, an Android context and logging configuration, amongst other things).
In Dagger land, things again get a tad verbose. The Component plays a pretty important role as it defines our injection targets and the objects we will be providing. There is also the provision of providing external dependencies to the Dagger graph (the application
class in this case).
To actually use it, we will need to build the component:
Clients
Now for the most interesting part! Here we will dive into what primarily differentiates Dagger and Koin and get to know a bit more about DI (and Service Locators too).
Constructor Injection
Let’s take a look at constructor injection first. With Koin, you simply write your constructors as you normally would. No need for any annotations:
However, in the case of Dagger, we see that we need to annotate the class with @Inject
. Also, we need to use the verbose variant of class constructor:
Is that it? Well, no. If the class had 3 more dependencies, in the case of Koin, we would have to find our module and convert:
module { viewModel { ProfileViewModel(get()) } } |
to
module { viewModel { ProfileViewModel(get(), get(), get(), get()) } } |
And if the dependencies of any of these 4 classes had changed, well, you guessed it—add the get()
for them as well. This does seem like a whole lot of manual work, and Jake Wharton certainly seems to agree when he tweeted this. And it does make sense. It can be easily seen how this can become untenable in big projects.
And how does Dagger fare in this regard? Well, it’s as easy as adding @Inject constructor(...)
to the class. This scales really well for bigger projects as you don’t have to go around hunting for modules and modify them.
Member Injection
When injecting members, Dagger and Koin provide very similar (but also very different 🙃) APIs.
Going with Koin first, we see that we use by
inject()
to inject members.
It is kind-of similar for Dagger, where we get a hold of the component and call inject
manually.
Looking at these examples, it seems that both Koin and Dagger seem to be acting as Service Locators. For a refresher, here is the difference between DI and SL as stated by Martin Fowler:
"With service locator the application class asks for it explicitly by a message to the locator. With injection there is no explicit request, the service appears in the application class - hence the inversion of control."
In both our examples, the Fragment is the one requesting the dependencies. So does that mean that Dagger isn’t a DI library? Well, no. If you stop looking at it from strictly an Android perspective, you will see that we can perform member injection from outside a class with Dagger but not for Koin. Let’s take a look at another example:
This is what I meant when I wrote that they have similar but different APIs. We are able to satisfy member dependencies from outside the class. In Android, we do use Dagger as a service locator, but that is only because of how the Activities and Fragments are designed. This is one of the primary confusions that I had, seeing how similar member injection is for both the libraries.
You might say, “OK, if I use constructor injection to a bare minimum, then I should be good with using Koin, right?” Ehhhh…. not really, no. In fact, you should be doing the opposite. Member injection should be minimized in Android. You should write as much of the code you can in constructor injected types, and only use member injection to integrate your code with the Android components.
Conclusion
I wrote this post with the intention of shedding more light on the definitions of service locator, dependency injection, and how Dagger and Koin fit those definitions.
Koin is definitely easier to learn and provides you with quicker build times. I have tried it in a few small to mid-sized projects and it has been a great experience using it. However, the lack of compile-time safety was missed; given that I am so used to it using Dagger for so long. Also, manual graph maintenance can start to become bothersome after a certain point. While there is no denying that learning Dagger is a monumental task and it does affect your build speeds, it does offer some sweet benefits. In the end, both Koin and Dagger are just tools. You will need to decide which one fits your needs and I hope this article will be at least of little help in deciding which one to go with!
If you are interested in learning more, check out these great resources which helped me develop a better understanding of the topics discussed in this article.