Предположим, у меня есть тот, 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 в конструктор.