Ninject Kernel.Get 和构造函数注入之间的不同行为
Ninject different behaviour between Kernel.Get and Constructor Injection
我有什么:
public interface IBla
{
}
public class Bla1 : IBla
{
}
public class Bla : IBla
{
}
public class Consumer
{
private readonly IBla[] _array;
public Consumer(IBla[] array)
{
_array = array;
}
}
public static class NinjectExtensions
{
public class BindListExpression<TElement>
{
private readonly IKernel _kernel;
private readonly List<Type> _types = new List<Type>();
public BindListExpression(IKernel kernel)
{
_kernel = kernel;
}
public BindListExpression<TElement> ImplementedBy<T>() where T : TElement
{
var type = typeof(T);
_kernel.Bind<T>().To(type);
_types.Add(type);
return this;
}
public void Bind()
{
Func<TElement[]> createObjects = () =>
{
var sourceArray = new TElement[_types.Count];
for (var i = 0; i < _types.Count; i++)
{
var value = _kernel.Get(_types[i]);
sourceArray[i] = (TElement)value;
}
return sourceArray;
};
_kernel.Bind<TElement[]>().ToMethod(x => createObjects().ToArray());
_kernel.Bind<List<TElement>>().ToMethod(x => (createObjects().ToList()));
_kernel.Bind<IEnumerable<TElement>>().ToMethod(x => createObjects().ToList());
}
}
public static BindListExpression<T> ListOf<T>(this IKernel kernel)
{
return new BindListExpression<T>(kernel);
}
}
用法:
// Binds items in the given order as a list (Ninject does not guarantee the given order so I use this mechanism).
kernel.ListOf<IBla>()
.ImplementedBy<Bla1>()
.ImplementedBy<Bla>()
.Bind();
var consumer = kernel.Get<Consumer>(); // result: consumer._array is empty?! --> what is imo wrong
var array = kernel.Get<IBla[]>(); // result: Bla1, Bla --> correct
为什么 Ninject 在 Get<IBla[]>()
和带有参数 IBla[]
的构造函数之间产生相同的结果?
通过构造函数注入,ninject 将 ctor 参数 IBla[]
转换为 IResolutionRoot.GetAll<IBla>().ToArray()
。这就是实现多注入支持的方式。因此,ctor-request 不可能导致 IResolutionRoot.Get<IBla[]>()
- 但它仍然是您可以手动执行的操作。
这适用于所有 ninject 转换为多重注入的集合类型(AFAIR 数组、IList
、IEnumerable
,但 ICollection
不适用)。
我建议使用另一个集合接口(如 ICollection
)或集合实现作为构造函数参数。这将导致 ctor 注入和 IResolutionRoot.Get
调用的一致行为。
可以按特定顺序绑定数组依赖项。你只需要像这样在Ninject
中注册它们。
_kernel.Bind<Bla1>().ToSelf();
_kernel.Bind<Bla>().ToSelf();
_kernel.Bind<IConsumer>().To<Consumer>()
.WithConstructorArgument("array",
new IBla[] {
_kernel.Get<Bla1>(),
_kernel.Get<Bla>()
});
我有什么:
public interface IBla
{
}
public class Bla1 : IBla
{
}
public class Bla : IBla
{
}
public class Consumer
{
private readonly IBla[] _array;
public Consumer(IBla[] array)
{
_array = array;
}
}
public static class NinjectExtensions
{
public class BindListExpression<TElement>
{
private readonly IKernel _kernel;
private readonly List<Type> _types = new List<Type>();
public BindListExpression(IKernel kernel)
{
_kernel = kernel;
}
public BindListExpression<TElement> ImplementedBy<T>() where T : TElement
{
var type = typeof(T);
_kernel.Bind<T>().To(type);
_types.Add(type);
return this;
}
public void Bind()
{
Func<TElement[]> createObjects = () =>
{
var sourceArray = new TElement[_types.Count];
for (var i = 0; i < _types.Count; i++)
{
var value = _kernel.Get(_types[i]);
sourceArray[i] = (TElement)value;
}
return sourceArray;
};
_kernel.Bind<TElement[]>().ToMethod(x => createObjects().ToArray());
_kernel.Bind<List<TElement>>().ToMethod(x => (createObjects().ToList()));
_kernel.Bind<IEnumerable<TElement>>().ToMethod(x => createObjects().ToList());
}
}
public static BindListExpression<T> ListOf<T>(this IKernel kernel)
{
return new BindListExpression<T>(kernel);
}
}
用法:
// Binds items in the given order as a list (Ninject does not guarantee the given order so I use this mechanism).
kernel.ListOf<IBla>()
.ImplementedBy<Bla1>()
.ImplementedBy<Bla>()
.Bind();
var consumer = kernel.Get<Consumer>(); // result: consumer._array is empty?! --> what is imo wrong
var array = kernel.Get<IBla[]>(); // result: Bla1, Bla --> correct
为什么 Ninject 在 Get<IBla[]>()
和带有参数 IBla[]
的构造函数之间产生相同的结果?
通过构造函数注入,ninject 将 ctor 参数 IBla[]
转换为 IResolutionRoot.GetAll<IBla>().ToArray()
。这就是实现多注入支持的方式。因此,ctor-request 不可能导致 IResolutionRoot.Get<IBla[]>()
- 但它仍然是您可以手动执行的操作。
这适用于所有 ninject 转换为多重注入的集合类型(AFAIR 数组、IList
、IEnumerable
,但 ICollection
不适用)。
我建议使用另一个集合接口(如 ICollection
)或集合实现作为构造函数参数。这将导致 ctor 注入和 IResolutionRoot.Get
调用的一致行为。
可以按特定顺序绑定数组依赖项。你只需要像这样在Ninject
中注册它们。
_kernel.Bind<Bla1>().ToSelf();
_kernel.Bind<Bla>().ToSelf();
_kernel.Bind<IConsumer>().To<Consumer>()
.WithConstructorArgument("array",
new IBla[] {
_kernel.Get<Bla1>(),
_kernel.Get<Bla>()
});