使用c#(propertyInfo)中的反射初始化IEnumerable属性

Sti*_*pen 2 c# reflection ienumerable

假设一个班级(可能是其他人)

class Foo()
{
     IEnumerable<int> SomeNumbers { get; set; }
     IEnumerable<string> SomeStrings { get; set; }       
     int[] ArrayOfInt { get; set; }
     List<int> AListOfIntegers { get; set; }
}
Run Code Online (Sandbox Code Playgroud)

和方法

void Initialize(object obj)
{
     var props = obj.getType().GetProperties();
     foreach(var prop in props)
     {
         //Some logic in case property is a single value, this is using:
         //Convert.ChangeType(value, property.PropertyType, null)

         var list = //Do Some Magic
         prop.SetValue(obj, list, null);
     }
}
Run Code Online (Sandbox Code Playgroud)

如何创建正确类型的列表?

我试过了:

var list = property.PropertyType.GetConstructor(new CType[0]).Invoke(new object[0]);
Run Code Online (Sandbox Code Playgroud)

但这不起作用,因为IEnumerable没有构造函数(它是一个接口),但任何超类型都可以.

var list = Enumerable.Empty<property.PropertyType.MemberType>();
Run Code Online (Sandbox Code Playgroud)

甚至不合法

And*_*yak 5

您可以使用以下代码创建List<T>该继承的实例IEnumerable<T>:

var genericType = property.PropertyType.GetGenericArguments().First();     
var instance = Activator.CreateInstance(typeof(List<>).MakeGenericType(genericType));
Run Code Online (Sandbox Code Playgroud)

结果instance可以设置为属性值.

prop.SetValue(obj, instance, null);
Run Code Online (Sandbox Code Playgroud)

您也可以Enumerable<T>使用以下代码创建空:

var genericType = property.PropertyType.GetGenericArguments().First();             
var method = typeof(Enumerable).GetMethod("Empty").MakeGenericMethod(genericType);
var emptyEnumerable = method.Invoke(null, null); 
Run Code Online (Sandbox Code Playgroud)