Suponha que eu tenha o Service
que recebe dependências via construtor, mas também precise ser inicializado com dados personalizados (contexto) antes de poder ser usado:
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;
}
Agora - os dados de contexto não são conhecidos de antemão, portanto, não posso registrá-los como uma dependência e usar o DI para injetá-los no serviço
É assim que o exemplo de cliente se parece:
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();
}
}
Como você pode ver - existem acoplamentos temporais e odores de código de método de inicialização envolvidos, porque primeiro eu preciso ligar service.Initialize
para poder ligar service.DoSomething
e service.DoOtherThing
depois.
Quais são as outras abordagens nas quais posso eliminar esses problemas?
Esclarecimentos adicionais sobre o comportamento:
Cada instância do cliente precisa ter sua própria instância do serviço inicializada com dados de contexto específicos do cliente. Portanto, esses dados de contexto não são estáticos ou são conhecidos antecipadamente, portanto, não podem ser injetados por DI no construtor.