Подтвердить что ты не робот

Найти все производные типы родового класса

У меня есть общий класс и производный класс следующим образом.

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<>));
4b9b3361

Ответ 1

var result = System.Reflection.Assembly.GetExecutingAssembly()
    .GetTypes()
    .Where(t => t.BaseType != null && t.BaseType.IsGenericType && 
                t.BaseType.GetGenericTypeDefinition() == typeof(GenericClass<>));

Ответ 2

Это немного сложнее, чем это. 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;
}

Ответ 3

Поскольку мне нужно было рекурсивно найти все производные типы, я написал этот код и поделился бы с любым, кто может понадобиться:

    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);
        }
    }