Ответ 1
var result = System.Reflection.Assembly.GetExecutingAssembly()
.GetTypes()
.Where(t => t.BaseType != null && t.BaseType.IsGenericType &&
t.BaseType.GetGenericTypeDefinition() == typeof(GenericClass<>));
У меня есть общий класс и производный класс следующим образом.
public class GenericClass<T> { ... }
public class DerivedClass : GenericClass<SomeType> { ... }
Как найти производный класс через отражение? Я пробовал оба способа ниже, но, похоже, не работает.
System.Reflection.Assembly.GetExecutingAssembly().GetTypes().Where(t => typeof(GenericClass<>).IsAssignableFrom(t));
System.Reflection.Assembly.GetExecutingAssembly().GetTypes().Where(t => t.IsSubclassOf(typeof(GenericClass<>));
var result = System.Reflection.Assembly.GetExecutingAssembly()
.GetTypes()
.Where(t => t.BaseType != null && t.BaseType.IsGenericType &&
t.BaseType.GetGenericTypeDefinition() == typeof(GenericClass<>));
Это немного сложнее, чем это. t.BaseType может возвращать нуль (например, когда t является интерфейсом). Также обратите внимание, что метод Type.IsSubclassOf не работает для общих типов! Если вы имеете дело с родовым типом, вы должны использовать метод GetTypeDefinition. Недавно я писал о как получить все производные типы класса. Здесь также используется метод IsSubclass, который также работает для дженериков:
public static bool IsSubclassOf(Type type, Type baseType)
{
if (type == null || baseType == null || type == baseType)
return false;
if (baseType.IsGenericType == false)
{
if (type.IsGenericType == false)
return type.IsSubclassOf(baseType);
}
else
{
baseType = baseType.GetGenericTypeDefinition();
}
type = type.BaseType;
Type objectType = typeof(object);
while (type != objectType && type != null)
{
Type curentType = type.IsGenericType ?
type.GetGenericTypeDefinition() : type;
if (curentType == baseType)
return true;
type = type.BaseType;
}
return false;
}
Поскольку мне нужно было рекурсивно найти все производные типы, я написал этот код и поделился бы с любым, кто может понадобиться:
public void ListAllDerviedTypes()
{
Type entityType = typeof(TableAdapter);
Assembly assembly = Assembly.LoadFrom(entityType.Assembly.Location);
Type[] types = assembly.GetTypes();
List<Type> results = new List<Type>();
GetAllDerivedTypesRecursively(types, typeof(SiteAndSectorsTable<>), ref results);
foreach (var type in results)
{
Console.WriteLine(type.Name);
}
}
private static void GetAllDerivedTypesRecursively(Type[] types, Type type1, ref List<Type> results)
{
if (type1.IsGenericType)
{
GetDerivedFromGeneric(types, type1, ref results);
}
else
{
GetDerivedFromNonGeneric(types, type1, ref results);
}
}
private static void GetDerivedFromGeneric(Type[] types, Type type, ref List<Type> results)
{
var derivedTypes = types
.Where(t => t.BaseType != null && t.BaseType.IsGenericType &&
t.BaseType.GetGenericTypeDefinition() == type).ToList();
results.AddRange(derivedTypes);
foreach (Type derivedType in derivedTypes)
{
GetAllDerivedTypesRecursively(types, derivedType, ref results);
}
}
public static void GetDerivedFromNonGeneric(Type[] types, Type type, ref List<Type> results)
{
var derivedTypes = types.Where(t => t != type && type.IsAssignableFrom(t)).ToList();
results.AddRange(derivedTypes);
foreach (Type derivedType in derivedTypes)
{
GetAllDerivedTypesRecursively(types, derivedType, ref results);
}
}