Припустимо, у мене є те, Service
що отримує залежності через конструктор, але його потрібно ініціалізувати спеціальними даними (контекстом), перш ніж вони можуть бути використані:
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;
}
Тепер - дані контексту заздалегідь невідомі, тому я не можу зареєструвати його як залежність і використовувати DI для введення його в службу
Ось як виглядає приклад клієнта:
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();
}
}
Як бачимо - тут пов'язані тимчасові з’єднання та ініціалізація коду методу запахів, тому що мені спочатку потрібно зателефонувати, service.Initialize
щоб мати можливість дзвонити service.DoSomething
і service.DoOtherThing
після цього.
Які ще є підходи, за допомогою яких я можу усунути ці проблеми?
Додаткове уточнення поведінки:
Кожен екземпляр клієнта повинен мати власний екземпляр послуги, ініціалізований конкретними контекстними даними клієнта. Отже, ці контекстні дані не є статичними або заздалегідь відомими, тому їх не можна вводити DI в конструктор.