Constructor Injection
In general, dependencies are constructed and injected when they're resolved for the first time (with some exceptions).
When registering a class that needs dependencies, provide a constructor that takes them as arguments. VContainer will take care of the rest.
note
At this time, optional dependencies are not supported in constructors. If a constructor dependency is missing, VContainer will throw an exception when building or validating a container.
Here's a basic example.
class ClassA{ readonly IServiceA serviceA; readonly IServiceB serviceB; readonly SomeUnityComponent component;
public ClassA( IServiceA serviceA, IServiceB serviceB, SomeUnityComponent component) { this.serviceA = serviceA; this.serviceB = serviceB; this.component = component; }}
The above code sample does not use VContainer APIs, because it doesn't need any. You can still manually create instances of ClassA
with the same constructor if you'd like.
caution
Unity strips unused code from builds, but it often makes mistakes when reflection or IL code generation are involved (as is the case with VContainer). To ensure your constructor is not removed, use a link.xml
file or add the [Inject]
attribute to one constructor. This also applies to methods.
[Inject] public ClassA( IServiceA serviceA, IServiceB serviceB, SomeUnityComponent component) { // ... }
note
If a class has more than one constructor, VContainer will throw an exception unless exactly one constructor defines [Inject]
. In such a case, VContainer prioritizes that one constructor when resolving dependencies.
Recommendation
Use constructors and readonly
fields for injection whenever possible. Here's why:
- It's the simplest injection form to use.
- If used with VContainer, the class can safely assume that its dependencies have been resolved (or else it wouldn't have been constructed).
- There's no magic in your class; you can even instantiate it without using VContainer (e.g. for testing).
- Class dependencies are made obvious. It also becomes easy to tell when a class has too many responsibilities (because its constructor will be too big).
MonoBehaviour
#
MonoBehaviour
s don't support constructors. Use method injection or property/field injection instead.
#
Suppressing WarningsUnity includes a set of code analysis attributes defined by JetBrains that some IDEs (including Rider) use for code analysis. If your IDE marks your injection constructor as unused, apply [UsedImplicitly]
(possibly with ImplicitUseKindFlags.InstantiatedNoFixedConstructorSignature
):
public class Dependency{ [UsedImplicitly(ImplicitUseKindFlags.InstantiatedNoFixedConstructorSignature)] public Dependency(GameContext game, StatusContext status) { _game = game; }}// This class and constructor will not be marked as unused, but the status parameter will be.
When constructors or methods are annotated as described, Rider will prevent the constructor (and the class) from being marked as unused. However, unused parameters within that constructor will still be marked as such.
Results may vary for other IDEs.