我想将Castle Windsor配置为创建两个相同类型的组件(Foo - > IFoo),但具有不同的构造函数输入。我后来也想在创建另一个组件时使用这两个组件(键入Bar - 请参阅下面的代码)。
public interface IFoo { }
public class Foo : IFoo
{
private string _prop;
public Foo(string prop)
{
_prop = prop;
}
}
public class Bar
{
private IFoo _fooAbc;
private IFoo _foo123;
public Bar(IFoo fooAbc, IFoo foo123)
{
_foo123 = foo123;
_fooAbc = fooAbc;
}
}
在组件安装程序中,我尝试注册这样的组件:
public void Install(IWindsorContainer container, IConfigurationStore store)
{
container.Register(Classes.FromThisAssembly()
.BasedOn<IFoo>().WithServiceBase()
.ConfigureFor<Foo>(c => c.DependsOn(Dependency.OnValue<string>("abc")).Named("fooAbc"))
.ConfigureFor<Foo>(c => c.DependsOn(Dependency.OnValue<string>("123")).Named("foo123")));
container.Register(Component.For<Bar>()); //?? specify which service to use
}
但是城堡引发了注册例外。那么如何配置两个Foo实例,一个带有“abc”,另一个带有“123”依赖?此外,我后来想在构造Bar时正确分配它们,以便fooAbc用作第一个构造函数输入,foo123用作第二个。我的最终目标是成功解决Bar。
答案 0 :(得分:1)
我不确定这是否更接近您的要求,但是,您可以使用
ServiceOverride.ForKey
指定哪些参数映射到哪些名称:
Component.For<Bar>().ImplementedBy<Bar>().
DependsOn(ServiceOverride.ForKey("fooAbc").Eq("abc")).
DependsOn(ServiceOverride.ForKey("foo123").Eq("123"))
);
或者,不是直接答案,但您可以选择解决IEnumerable<IFoo>
。如果您实际有任意数量的IFoo
要解决,这是一个不错的选择。
如果您更改Bar的定义以接受IEnumerable
public class Bar
{
private readonly IEnumerable<IFoo> _foos;
public Bar(IEnumerable<IFoo> foos)
{
_foos = foos;
}
}
然后注册并解决。在进行注册之前,您需要添加Resolve。
var container = new WindsorContainer();
container.Kernel.Resolver.AddSubResolver(new CollectionResolver(container.Kernel, true));
container.Register(
Component.For<IFoo>().Instance(new Foo("abc")).Named("abc"),
Component.For<IFoo>().Instance(new Foo("123")).Named("123"),
Component.For<Bar>().ImplementedBy<Bar>());