我正在尝试构建一个DI容器,我偶然发现了以下问题:我有一个方法可以检索给定类型的已注册实例列表,我想用它来注入{{1}给定对象中的属性。我想要实现的一个例子如下:
IEnumerable<T>
我的class A { public IList<IExample> Objects { get; set; } }
class B: IExample {}
class C: IExample {}
Container.Register<IExample>(new B());
Container.Register<IExample>(new C());
var obj = new A();
Container.Inject(A);
Debug.Assert(A.Objects != null && A.Objects.Count == 2);
方法返回Retrieve
,主要是因为我当时没有类型信息,因此我尝试在注入时将该列表转换为IList<object>
。这是执行工作的方法的一种简单形式:
List<T>
代码返回错误:public virtual IList<object> Retrieve(Type type)
{
var instances = Registry[type];
foreach(var instance in instances)
Inject(type, instance); // omitted
return instances;
}
public virtual void Inject<T>(T instance)
{
var properties = typeof (T).GetProperties();
foreach (var propertyInfo in properties)
{
var propertyType = propertyInfo.PropertyType;
if (!IsIEnumerable(propertyType)) continue;
var genericType = propertyType.GetGenericArguments()[0];
propertyInfo.SetValue(instance,
GetListType(genericType, Retrieve(genericType)), null);
}
}
protected virtual object GetListType(Type type, IEnumerable<object> items)
{
return items.Select(item => Convert.ChangeType(item, type)).ToList();
}
遗憾的是,我不知道如何从这里开始。也许我这样做是错的。我曾想过使用泛型或手工注入多个属性,但我真的不想这样做。
提前感谢任何帮助或想法。
答案 0 :(得分:7)
您可以创建如下通用列表:
public virtual IList Retrieve(Type type)
{
// ...
listType = typeof(List<>).MakeGenericType(new Type[] { type });
IList list = (IList)Activator.CreateInstance(listType);
// ...
return list
}
此列表可以转换为IList<T>
,因为它是一个。
您可以考虑使用IEnumerable
和Cast<T>
,但之后您没有列表实例。我不知道拥有它是多么重要。