Angenommen, ich habe die Service
, die Abhängigkeiten über den Konstruktor empfängt, aber auch mit benutzerdefinierten Daten (Kontext) initialisiert werden muss, bevor sie verwendet werden können:
public interface IService
{
void Initialize(Context context);
void DoSomething();
void DoOtherThing();
}
public class Service : IService
{
private readonly object dependency1;
private readonly object dependency2;
private readonly object dependency3;
public Service(
object dependency1,
object dependency2,
object dependency3)
{
this.dependency1 = dependency1 ?? throw new ArgumentNullException(nameof(dependency1));
this.dependency2 = dependency2 ?? throw new ArgumentNullException(nameof(dependency2));
this.dependency3 = dependency3 ?? throw new ArgumentNullException(nameof(dependency3));
}
public void Initialize(Context context)
{
// Initialize state based on context
// Heavy, long running operation
}
public void DoSomething()
{
// ...
}
public void DoOtherThing()
{
// ...
}
}
public class Context
{
public int Value1;
public string Value2;
public string Value3;
}
Jetzt - die Kontextdaten sind nicht im Voraus bekannt, daher kann ich sie nicht als Abhängigkeit registrieren und DI verwenden, um sie in den Dienst einzufügen
So sieht ein Beispielclient aus:
public class Client
{
private readonly IService service;
public Client(IService service)
{
this.service = service ?? throw new ArgumentNullException(nameof(service));
}
public void OnStartup()
{
service.Initialize(new Context
{
Value1 = 123,
Value2 = "my data",
Value3 = "abcd"
});
}
public void Execute()
{
service.DoSomething();
service.DoOtherThing();
}
}
Wie Sie sehen können, handelt es sich um zeitliche Kopplungs- und Initialisierungsgerüche von Methodencode, da ich zuerst aufrufen muss, um aufrufen service.Initialize
zu können, service.DoSomething
und service.DoOtherThing
danach.
Was sind die anderen Ansätze, mit denen ich diese Probleme beseitigen kann?
Zusätzliche Klarstellung des Verhaltens:
Jede Instanz des Clients muss über eine eigene Instanz des Dienstes verfügen, die mit den spezifischen Kontextdaten des Clients initialisiert wurde. Diese Kontextdaten sind also nicht statisch oder im Voraus bekannt, sodass sie nicht von DI in den Konstruktor eingefügt werden können.