WCF 支持构造函数注入,但您必须跳过几圈才能到达那里。关键在于编写一个自定义的 ServiceHostFactory。虽然它也必须有一个默认构造函数,但您可以使用它来连接所有正确的行为。
例如,我最近编写了一个使用 Castle Windsor 来连接服务实现的依赖项的示例。 CreateServiceHost 的实现很简单:
return new WindsorServiceHost(this.container, serviceType, baseAddresses);
其中this.container 是已配置的 IWindsorContainer。
WindsorServiceHost 如下所示:
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));
}
}
}
WindsorInstanceProvider 看起来像这样:
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
}
这可能看起来很多,但请注意,它是可重用的通用代码,具有相当低的圈复杂度。
您可以按照相同的编码习惯使用另一个 DI 容器或使用可怜人的 DI 来实现依赖注入。
这是使用Poor Man's DI的这个成语的older writeup。