Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Strategy Pattern and Dependency Injection using Unity

Tags:

I am finally getting my feet wet with Dependency Injection (long overdue); I got started playing with Unity and run into an issue with the strategy pattern. I can use the container to return to me specific implementations of a strategy based on a name, but what I don't see is how I am supposed to get the right strategy in the context.
Let's illustrate on a simple example: the context is a car, which has an IEngine (the strategy), with 2 implementations, FastEngine and SlowEngine. The code would look along these lines:

public interface IEngine {     double MaxSpeed     {         get;     } }  internal class FastEngine:IEngine {     public double MaxSpeed     {         get          {              return 100d;          }     } }  internal class SlowEngine:IEngine {     public double MaxSpeed     {         get         {             return 10d;         }     } }  public class Car {     private IEngine engine;     public double MaximumSpeed     {         get         {             return this.engine.MaxSpeed;         }     }      public Car(IEngine engine)     {         this.engine = engine;     } } 

My problem is the following: how should I go about instantiating a fast car or a slow car? I can use the container to provide me with each implementation, and I can set a "default" implementation to use:

IUnityContainer container = new UnityContainer(); container.RegisterType<IEngine, FastEngine>(); container.RegisterType<IEngine, FastEngine>("Fast"); container.RegisterType<IEngine, SlowEngine>( "Slow" ); var car = container.Resolve<Car>(); Assert.AreEqual(100, car.MaximumSpeed); 

but what I would like is to be able to request a car with a specific implementation of the strategy - something like

var car = container.Resolve<Car>(??? use "Fast" or "Slow ???); 

Can I use the container to do that? Or should I write a Factory which uses the container? Any guidance would be appreciated - I am not sure I am thinking right about this!

like image 525
Mathias Avatar asked Nov 10 '09 06:11

Mathias


People also ask

Can you use dependency injection in unity?

Scene-based dependency injection means you can setup up the Unity hierarchy and your MonoBehaviors however you like. It scans your scene and automatically wires together your MonoBehaviors. It allows you to connect objects in the Unity hierarchy without having to manually use functions such as FindObjectOfType(...) .

Is dependency injection a strategy pattern?

Dependency injection is a refinement of the strategy pattern which I will briefly explain. It is often necessary to choose between several alternative modules at runtime. These modules all implement a common interface so that they can be used interchangeably.

What is unity config MVC?

UnityConfig is a static class which has a static method called RegisterComponents. Inside of this method we need to register all required dependencies. Registering Dependencies: Open UnityConfig. cs file from App_Start, inside of RegisterComponents method all dependency need to be registered.


1 Answers

A common pattern in DI is that at run-time there's only going to be a single implementation of a given abstraction. That just makes life a whole lot easier, as you don't need to deal with the ambiguity such as the one you describe.

However, sometimes, you need to vary an implementation based on context, such as the example you give. Many DI Containers provide ways where you can provide a qualifying parameter, but that means that you will end up tightly coupling your code to a specific DI Container.

A much better solution would be to introduct an Abstract Factory that can provide what you need. Something like

public interface ICarFactory {     Car Create(IEngine engine); } 

If you need to inject more Strategies, perhaps the Builder design pattern might fit even better.

In any case, the point is that instead of registering a lot of different Cars in the container, you would instead register a single ICarFactory implementation.

In your client code, you would use the injected ICarFactory to create a Car instance based on a particular IEngine.

var car = factory.Create(engine); 
like image 126
Mark Seemann Avatar answered Oct 27 '22 08:10

Mark Seemann