WCF unterstützt Konstruktor Injection, aber Sie müssen durch ein paar Reifen springen, um dorthin zu gelangen. Der Schlüssel liegt darin, eine benutzerdefinierte ServiceHostFactory zu schreiben. Auch wenn dies einen Standardkonstruktor haben muss, können Sie damit alle korrekten Verhaltensweisen verbinden.
Als Beispiel habe ich kürzlich eine geschrieben, die Castle Windsor verwendet, um Abhängigkeiten für die Service-Implementierung zu verkabeln. Die Implementierung von CreateServiceHost einfach tut dies:
return new WindsorServiceHost(this.container, serviceType, baseAddresses);
wo this.container
ist eine konfigurierte IWindsorContainer.
WindsorServiceHost sieht wie folgt aus:
public class WindsorServiceHost : ServiceHost
{
public WindsorServiceHost(IWindsorContainer container, Type serviceType, params Uri[] baseAddresses)
: base(serviceType, baseAddresses)
{
if (container == null)
{
throw new ArgumentNullException("container");
}
foreach (var cd in this.ImplementedContracts.Values)
{
cd.Behaviors.Add(new WindsorInstanceProvider(container));
}
}
}
und WindsorInstanceProvider sieht wie folgt aus:
public class WindsorInstanceProvider : IInstanceProvider, IContractBehavior
{
private readonly IWindsorContainer container;
public WindsorInstanceProvider(IWindsorContainer container)
{
if (container == null)
{
throw new ArgumentNullException("container");
}
this.container = container;
}
#region IInstanceProvider Members
public object GetInstance(InstanceContext instanceContext, Message message)
{
return this.GetInstance(instanceContext);
}
public object GetInstance(InstanceContext instanceContext)
{
var serviceType = instanceContext.Host.Description.ServiceType;
return this.container.Resolve(serviceType);
}
public void ReleaseInstance(InstanceContext instanceContext, object instance)
{
this.container.Release(instance);
}
#endregion
#region IContractBehavior Members
public void AddBindingParameters(ContractDescription contractDescription, ServiceEndpoint endpoint, BindingParameterCollection bindingParameters)
{
}
public void ApplyClientBehavior(ContractDescription contractDescription, ServiceEndpoint endpoint, ClientRuntime clientRuntime)
{
}
public void ApplyDispatchBehavior(ContractDescription contractDescription, ServiceEndpoint endpoint, DispatchRuntime dispatchRuntime)
{
dispatchRuntime.InstanceProvider = this;
}
public void Validate(ContractDescription contractDescription, ServiceEndpoint endpoint)
{
}
#endregion
}
Dieses wie viel aussehen, aber feststellen, dass es wiederverwendbar, Allzweck-Code ist, dass ein eher hat geringe zyklomatische Komplexität.
Sie können dieselbe Codierungssprache verwenden, um Dependency Injection mit einem anderen DI-Container zu implementieren, oder indem Sie DI von Poor Man verwenden.
Hier ist ein older writeup dieser Idiom, die Armer Menschen DI verwendet.
verwenden Sie einen IoC-Container? und wenn ja welche? – Fabiano
Ich verwende noch keinen IoC-Container auf der Serverseite. Ich plane jetzt eine einzuführen. Ich verwende Unity auf der Client-Seite (wie es mit Prism kam), aber erwäge, StructureMap auf dem Server zu verwenden. Offen für jeden wirklich. – stiank81