Say I have a class Util that takes in a object - an instance of class Validator.
Since I want to avoid instantiating the Validator class within Util, I pass it in via a constructor:
public class Util { @Inject public Util(Validator validator) { } }
I have a module that provides the Validator instance:
@Provides @Singleton Validator provideValidator() { return Validator.getInstance(); }
and an instance of the Util class:
@Provides Util provideUtil(Validator validator) { return new Util(validator); }
I have a component wired up that would give me an instance of Util:
Util getUtil()
so within my activity, I could call it like:
Util myUtil = getComponent.getUtil();
All of that works fine - myUtil has a proper instance of Validator class when instantiated.
Now I want to pass in a String variable named address (which is user input via a UI). I want to change the constructor so I pass in both an instance of Validator and the user inputted String:
@Inject public Util(Validator validator, String address) { }
I just can't get my head around how to pass that 2nd parameter. Can someone tell me how?
Ideally, I want to instantiate Util like:
Util myUtil = getComponent.getUtil(txtAddress.getText());
Dagger 2 Components are the counterpart to Guice Injectors so the way to do this in Dagger 2 would be to specify the object whose field you want to inject at runtime as an injection site and request injection from the component.
With the @Inject annotation on the constructor, we instruct Dagger that an object of this class can be injected into other objects. Dagger automatically calls this constructor, if an instance of this class is requested.
Assisted injection is a dependency injection (DI) pattern that is used to construct an object where some parameters may be provided by the DI framework and others must be passed in at creation time (a.k.a “assisted”) by the user.
Dagger 2 is a compile-time android dependency injection framework that uses Java Specification Request 330 and Annotations. Some of the basic annotations that are used in dagger 2 are: @Module This annotation is used over the class which is used to construct objects and provide the dependencies.
Dagger 2 Dagger 2 is a compile-time android dependency injection framework that uses Java Specification Request 330 and Annotations. Some of the basic annotations that are used in dagger 2 are: @Module This annotation is used over the class which is used to construct objects and provide the dependencies.
This implementation of ViewModelFactory is safe to use: Now, instead of injecting ViewModels themselves into ViewModelFactory, I inject Providers of the respective ViewModels. Providers are like proxies into Dagger’s internal construction logic for a single type.
Now, to add new ViewModel, you’ll simply add the respective provider method: Dagger will automatically put Provider for ViewModel2 into Map and ViewModelFactory will be able to use it. The benefit of this method is clear: it reduces the amount of boilerplate that you need to write.
I had the same question as you when I started looking into Dagger 2 a couple of weeks ago. I found information on this (and most other Dagger 2-related issues) difficult to come by, so I hope this helps!
The most basic answer is that you cannot. What you are looking for is something that is called assisted injection, and it is not part of Dagger 2. Some other dependency injection (DI) frameworks, such as Guice, do offer this feature, so you might look into those. Of course, there are still ways to do what you want to do using Dagger 2.
The standard way to do what you want to do in combination with DI is by using the Factory pattern. Basically, you create an injectable factory class that takes runtime parameters such as address
as arguments to the object creation methods that it provides.
In your case, you would need a UtilFactory
into which Dagger 2 injects a Validator
upon instantation and which offers a method create(String address)
that creates instances of Util
. UtilFactory
should keep a reference to the injected instance of Validator
so that it has everything it needs to create an instance of Util
in the create
method.
Wring code for many such factories can be cumbersome. You should definitely take a look at AutoFactory, which eases some of the burden. Guice's assisted injection seems to work quite similar to Dagger 2 + AutoFactory (albeit with even nicer syntactic sugar).
I doubt this is something that you would like to do in this case, but you could just create a module that provides the address (and instantiate a new component). You do not have to create a new @Module class for every possible address. Instead, you can just pass the address as an argument to the constructor of the module. You could use the @BindsInstance-annotation as suggested by teano to achieve a similar result.
I am not sure if this is an anti-pattern or not. To me, this seems like an acceptable route in some cases, but only when you are actually using the same e.g. address for the initialisation of "many" objects. You definitely do not want to instantiate a new component and a new model for each object that requires injection. It is not efficient, and if you are not careful you will end up with more boilerplate code than without Dagger.
Something that was immensely useful to me when learning about DI frameworks was the realisation that using a DI framework does not mean that you have to DI to initialise all of your objects. As a rule of thumb: inject objects that you know of at compile time and that have static relations to other objects; do not inject runtime information.
I think this is a good post on the subject. It introduces the concept of 'newables' and 'injectables'.
Integer
, Address
etc. are examples of newables.Broadly speaking, newables are passive objects, and there is no point in injecting or mocking them. They typically contain the "data" that is in your application and that is only available at runtime (e.g. your address). Newables should not keep references to injectables or vice versa (something the author of the post refers to as "injectable/newable-separation").
In reality, I have found that it is not always easy or possible to make a clear distinction between injectables and newables. Still, I think that they are nice concepts to use as part of your thinking process. Definitely think twice before adding yet another factory to your project!
In your case, I think it would make sense to treat Util
as an injectable and the address as a newable. This means that the address should not be part of the Util
class. If you want to use instance of Util
for e.g. validating/... addresses, just pass the address that you want to validate as an argument to the validation/... method.
Since the 2.31 version of Dagger 2, there is also a mechanism for assisted injection using @AssistedInject
. You can see more in the documentation here. (Edited at the suggestion of Jay Sidri.)
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With