typeof(Upload).GetProperties()
.Where(p => typeof(UploadObject).IsAssignableFrom(p.PropertyType));
将返回返回类型为 UploadClass 或继承类的所有属性。然后,您将对其使用递归,并在不可能包含 UploadObject 的类型上使用停止条件。停止条件可以是使用IsPrimitive
(available on Type
) 检查返回类型是否为原始类型。
== typeof(UploadObject)
如果您只想要特定类型,则可以替换 IsAssignableFrom 。
编辑:
这是我心中的想法。免责声明:我所做的唯一测试是编译它并针对一个场景运行它,它适用于它,但它说明了这个想法。
public static T[] GetAllPropertyValuesForTypeinTree<T>(object rootObject)
{
var propertyInfos = rootObject.GetType().GetProperties();
var withCorrectPropertyType = propertyInfos
.Where(p => p.PropertyType == typeof(T)).ToList();
var withListsOfType = propertyInfos.Where(p =>
IsEnumerableOf<T>(p.PropertyType)).ToList();
var complexProperties = propertyInfos.Except(
withCorrectPropertyType.Concat(withListsOfType))
.Where(p => !p.PropertyType.IsPrimitive && !p.PropertyType.IsGenericType);
var complexValues = new List<T>();
foreach (var complexProperty in complexProperties)
{
complexValues.AddRange(GetAllPropertyValuesForTypeinTree<T>(
complexProperty.GetValue(rootObject, null)));
}
var listValues = withListsOfType.Select(p =>
(IEnumerable)p.GetValue(rootObject, null))
.SelectMany(p => p.OfType<T>());
var propertyValues = withCorrectPropertyType.Select(p =>
(T) p.GetValue(rootObject, null));
return propertyValues.Concat(listValues).Concat(complexValues).ToArray();
}
private static bool IsEnumerableOf<T>(Type type)
{
if (!typeof (IEnumerable).IsAssignableFrom(type))
return false;
return type.GetInterfaces().Any(interfaceType => interfaceType.IsGenericType
&& (interfaceType.GetGenericTypeDefinition() == typeof(IEnumerable<>)
&& type.GetGenericArguments()[0] == typeof(T)))
|| type == typeof(IEnumerable<T>);
}
你可以这样称呼它:
var propVals = GetAllPropertyValuesForTypeinTree<UploadObject>(theInstance);