40

如何通过 C# 中的反射获得接口的所有实现?

4

6 回答 6

59

答案是这样的;它搜索整个应用程序域——即应用程序当前加载的每个程序集。

/// <summary>
/// Returns all types in the current AppDomain implementing the interface or inheriting the type. 
/// </summary>
public static IEnumerable<Type> TypesImplementingInterface(Type desiredType)
{
    return AppDomain
           .CurrentDomain
           .GetAssemblies()
           .SelectMany(assembly => assembly.GetTypes())
           .Where(type => desiredType.IsAssignableFrom(type));
}

它是这样使用的;

var disposableTypes =  TypesImplementingInterface(typeof(IDisposable));

您可能还希望此函数查找实际的具体类型——即过滤掉抽象、接口和泛型类型定义。

public static bool IsRealClass(Type testType)
{
    return testType.IsAbstract == false
         && testType.IsGenericTypeDefinition == false
         && testType.IsInterface == false;
}
于 2008-09-17T06:11:51.710 回答
5

看看Assembly.GetTypes()方法。它返回可以在程序集中找到的所有类型。您所要做的就是遍历每个返回的类型并检查它是否实现了必要的接口。

这样做的方法是使用Type.IsAssignableFrom方法。

这是示例。myInterface是您正在搜索的接口和实现。

Assembly myAssembly;
Type myInterface;
foreach (Type type in myAssembly.GetTypes())
{
    if (myInterface.IsAssignableFrom(type))
        Console.WriteLine(type.FullName);
}

我确实相信这不是解决您的问题的一种非常有效的方法,但至少,这是一个很好的起点。

于 2008-09-17T05:43:42.183 回答
4
Assembly assembly = Assembly.GetExecutingAssembly();
List<Type> types = assembly.GetTypes();
List<Type> childTypes = new List<Type>();
foreach (Type type in Types) {
  foreach (Type interfaceType in type.GetInterfaces()) {
       if (interfaceType.Equals(typeof([yourinterfacetype)) {
            childTypes.Add(type)
            break;
       }
  }
}

也许是这样的......

于 2008-09-17T05:49:54.040 回答
3

正如Simon Farrow所建议的,这里有一些Type可能对此有用的扩展方法。此代码只是对已接受答案的重组。

代码

/// <summary>
/// Returns all types in <paramref name="assembliesToSearch"/> that directly or indirectly implement or inherit from the given type. 
/// </summary>
public static IEnumerable<Type> GetImplementors(this Type abstractType, params Assembly[] assembliesToSearch)
{
    var typesInAssemblies = assembliesToSearch.SelectMany(assembly => assembly.GetTypes());
    return typesInAssemblies.Where(abstractType.IsAssignableFrom);
}

/// <summary>
/// Returns the results of <see cref="GetImplementors"/> that match <see cref="IsInstantiable"/>.
/// </summary>
public static IEnumerable<Type> GetInstantiableImplementors(this Type abstractType, params Assembly[] assembliesToSearch)
{
    var implementors = abstractType.GetImplementors(assembliesToSearch);
    return implementors.Where(IsInstantiable);
}

/// <summary>
/// Determines whether <paramref name="type"/> is a concrete, non-open-generic type.
/// </summary>
public static bool IsInstantiable(this Type type)
{
    return !(type.IsAbstract || type.IsGenericTypeDefinition || type.IsInterface);
}

例子

要在调用程序集中获取可实例化的实现者:

var callingAssembly = Assembly.GetCallingAssembly();
var httpModules = typeof(IHttpModule).GetInstantiableImplementors(callingAssembly);

获取当前 AppDomain 中的实现者:

var appDomainAssemblies = AppDomain.CurrentDomain.GetAssemblies();
var httpModules = typeof(IHttpModule).GetImplementors(appDomainAssemblies);
于 2013-06-24T02:43:23.617 回答
1

您必须遍历您感兴趣的所有程序集。从程序集中您可以获得它定义的所有类型。请注意,当您执行 AppDomain.CurrentDomain.Assemblies 时,您只会获得已加载的程序集。直到需要时才会加载程序集,这意味着您必须在开始搜索之前显式加载程序集。

于 2008-09-17T10:12:29.930 回答
1

您的意思是 Type 实现的所有接口吗?

像这样:

ObjX foo = new ObjX();
Type tFoo = foo.GetType();
Type[] tFooInterfaces = tFoo.GetInterfaces();
foreach(Type tInterface in tFooInterfaces)
{
  // do something with it
}

希望有帮助。

于 2008-09-17T05:36:34.337 回答