如何确定类型是否使用C#反射实现接口


561

是否反映C#报价的方式来确定是否给予一些System.Type款型的一些接口?

public interface IMyInterface {}

public class MyType : IMyInterface {}

// should yield 'true'
typeof(MyType)./* ????? */MODELS_INTERFACE(IMyInterface);

Answers:


968

您有几种选择:

  1. typeof(IMyInterface).IsAssignableFrom(typeof(MyType))

  2. typeof(MyType).GetInterfaces().Contains(typeof(IMyInterface))

对于通用接口,则有所不同。

typeof(MyType).GetInterfaces().Any(i => i.IsGenericType && i.GetGenericTypeDefinition() == typeof(IMyInterface<>))

68
请记住,typeof(IMyInterface).IsAssignableFrom(typeof(IMyInterface))也为true,这可能会对您的代码产生意外结果。
克里斯·坎普

29
很容易不理会并IsAssignableFrom倒退。我GetInterfaces现在就去:p
本杰明·

12
IsAssignableFrom(t1)变体比GetInterfaces().Contains(t2)我的代码中的变体快约3倍。
皮埃尔·阿诺

24
@PierreArnaud:IsAssignableFrom最终会调用GetInterfaces,因此您的测试可能首先检查了GetInterfaces,之后检查了IsAssignable。那是因为GetInterfaces缓存了它的结果,所以第一次调用的成本更高
Panos Theof

17
@Kosta的答案有一个小的变化。使用C#6,我们可以实现typeof(MyType).GetInterface(nameof(IMyInterface)) != null更好的类型安全性和重构。
aholmes'3


32
typeof(IMyInterface).IsAssignableFrom(someclass.GetType());

要么

typeof(IMyInterface).IsAssignableFrom(typeof(MyType));

34
如果您已经有该类的实例,那么更好的方法就是根本someclass is IMyInterface不涉及反射成本。因此,虽然没有错,但这并不是实现它的理想方法。
詹姆士·雷根四世

1
@詹姆斯-同意。甚至Resharper也给出了相同的建议。
Angshuman Agarwal

@ JamesJ.ReganIV,您应该将其发布为答案,我几乎错过了您的评论
reggaeguitar 2014年

@reggaeguitar,谢谢,但评论未回答原始问题。问题要求提供反射解决方案,我只是说在此答案的第一种情况下,您确实拥有对象反射实例不是理想的解决方案。
詹姆斯·里根四世

1
@ JamesJ.ReganIV实际上,is在继承层次结构的两个方向都进行检查,而IsAssignableFrom仅向上进行检查。另外,如果您有一个对象的实例,则应调用IsInstanceOfType(它也只向上看)。
Sellorio '17

13
public static bool ImplementsInterface(this Type type, Type ifaceType) 
{
    Type[] intf = type.GetInterfaces();
    for(int i = 0; i < intf.Length; i++) 
    {
        if(intf[ i ] == ifaceType) 
        {
            return true;
        }
    }
    return false;
}

我认为这是正确的版本,原因有以下三个:

  1. 它使用GetInterfaces而不是IsAssignableFrom,它的速度更快,因为IsAssignableFrom最终在多次检查后都会调用GetInterfaces。
  2. 它遍历本地数组,因此不会进行边界检查。
  3. 它使用为Type定义的==运算符,因此可能比Equals方法(包含调用最终将使用)更安全。

10
内容+1,但我讨厌括号和埃及括号周围的空格。整个方法也可以写成:return type.GetInterfaces()。Any(t => t == ifaceType);
reggaeguitar 2014年

1
Type.IsAssignableFrom()内部行为与您的代码完全相同
devi 2014年

1
同样为什么不键入不使用LINQ的type.GetInterfaces()。Contains(ifaceType)。

9

我已经做了:

public static bool Implements<I>(this Type source) where I : class
{
  return typeof(I).IsAssignableFrom(source);
}

我希望我可以说过where I : interface,但interface不是通用参数约束选项。class尽可能地接近

用法:

if(MyType.Implements<IInitializable>())
  MyCollection.Initialize();

我刚才说的Implements是因为这样更直观。我总是被IsAssignableFrom人为打翻。


您可以return typeof(I).IsInterface && typeof(I).IsAssignableFrom(source);对方法的任何“不正确”用法返回false,即;将其与类类型而不是接口类型一起使用,或者,如果类型参数不是接口,则抛出异常。尽管您可能会争辩说派生类是“实现”的父类……
SindriJóelsson18年

7

修改Jeff的答案以获得最佳性能(感谢Pierre Arnaud的性能测试):

var type = typeof(MyType);
var implementsInterface = typeof(IMyInterface).IsAssignableFrom(type) && type.IsClass;

查找在给定中实现接口的所有类型Assembly

var implementations = typeof(TypeInTargetAssembly).Assembly.GetTypes()
                          .Where(t => typeof(IMyInterface).IsAssignableFrom(t) && t.IsClass);

7

正如其他人已经提到的:本杰明·13年4月10日在22:21”

可以很容易地不注意并向后获取IsAssignableFrom的参数。我现在就使用GetInterfaces:p –

好吧,另一种解决方法是创建一个简短的扩展方法,在某种程度上满足“最通常”的思维方式(并同意这是一个很小的个人选择,可以根据自己的喜好使其稍微“自然”些) ):

public static class TypeExtensions
{
    public static bool IsAssignableTo(this Type type, Type assignableType)
    {
        return assignableType.IsAssignableFrom(type);
    }
}

为什么不更通用一些(不知道是否真的那么有趣,我想我只是在传递另一撮“语法化”糖):

public static class TypeExtensions
{
    public static bool IsAssignableTo(this Type type, Type assignableType)
    {
        return assignableType.IsAssignableFrom(type);
    }

    public static bool IsAssignableTo<TAssignable>(this Type type)
    {
        return IsAssignableTo(type, typeof(TAssignable));
    }
}

我认为这样可能更自然,但再次只是个人意见的问题:

var isTrue = michelleType.IsAssignableTo<IMaBelle>();

4
您是否有理由不只是将实现直接放在扩展方法中?我的意思是肯定可以让您同时使用这两种方式,但是为什么您需要这样做呢?
Mark A. Donohoe

@MarqueIV很抱歉在迟到将近2年后再回覆您,好吧,我想这是一个古老的坏习惯,然后在扩展方法中包装辅助方法以避免重复代码,将编辑我的答案:)
Kerry Perret

1
@MarqueIV完成并改变了我不使用别名的另一个坏习惯,即Boolean=> bool(我不知道为什么我年轻时曾经有一些严格的“花哨”编码规则)。
凯瑞·佩雷

3

如果您有类型或实例,则可以轻松检查它们是否支持特定的接口。

要测试对象是否实现某个接口:

if(myObject is IMyInterface) {
  // object myObject implements IMyInterface
}

要测试一种类型是否实现了某个接口:

if(typeof(IMyInterface).IsAssignableFrom(typeof(MyType))) {
  // type MyType implements IMyInterface
}

如果您有一个通用对象,并且想要进行转换以及检查您转换到的接口是否已实现,则代码为:

 var myCastedObject = myObject as IMyInterface;

    if(myCastedObject != null) {
      // object myObject implements IMyInterface
    }

2

IsAssignableFrom现在移至TypeInfo

typeof(ISMSRequest).GetTypeInfo().IsAssignableFrom(typeof(T).GetTypeInfo());

1

任何搜索此内容的人都可能会发现以下扩展方法很有用:

public static class TypeExtensions
{
    public static bool ImplementsInterface(this Type type, Type @interface)
    {
        if (type == null)
        {
            throw new ArgumentNullException(nameof(type));
        }

        if (@interface == null)
        {
            throw new ArgumentNullException(nameof(@interface));
        }

        var interfaces = type.GetInterfaces();
        if (@interface.IsGenericTypeDefinition)
        {
            foreach (var item in interfaces)
            {
                if (item.IsConstructedGenericType && item.GetGenericTypeDefinition() == @interface)
                {
                    return true;
                }
            }
        }
        else
        {
            foreach (var item in interfaces)
            {
                if (item == @interface)
                {
                    return true;
                }
            }
        }

        return false;
    }
}

xunit测试:

public class TypeExtensionTests
{
    [Theory]
    [InlineData(typeof(string), typeof(IList<int>), false)]
    [InlineData(typeof(List<>), typeof(IList<int>), false)]
    [InlineData(typeof(List<>), typeof(IList<>), true)]
    [InlineData(typeof(List<int>), typeof(IList<>), true)]
    [InlineData(typeof(List<int>), typeof(IList<int>), true)]
    [InlineData(typeof(List<int>), typeof(IList<string>), false)]
    public void ValidateTypeImplementsInterface(Type type, Type @interface, bool expect)
    {
        var output = type.ImplementsInterface(@interface);
        Assert.Equal(expect, output);
    }
}

0

关于什么

if(MyType as IMyInterface != null)


4
当我有一个实例时,这是显而易见的。当我具有反射类型时,此功能
无用

0

关于什么

typeof(IWhatever).GetTypeInfo().IsInterface

0

正确答案是

typeof(MyType).GetInterface(nameof(IMyInterface)) != null;

然而,

typeof(MyType).IsAssignableFrom(typeof(IMyInterface));

可能会返回错误的结果,如以下代码所示,字符串和IConvertible:

    static void TestIConvertible()
    {
        string test = "test";
        Type stringType = typeof(string); // or test.GetType();

        bool isConvertibleDirect = test is IConvertible;
        bool isConvertibleTypeAssignable = stringType.IsAssignableFrom(typeof(IConvertible));
        bool isConvertibleHasInterface = stringType.GetInterface(nameof(IConvertible)) != null;

        Console.WriteLine($"isConvertibleDirect: {isConvertibleDirect}");
        Console.WriteLine($"isConvertibleTypeAssignable: {isConvertibleTypeAssignable}");
        Console.WriteLine($"isConvertibleHasInterface: {isConvertibleHasInterface}");
    }

结果:

 isConvertibleDirect: True
 isConvertibleTypeAssignable: False
 isConvertibleHasInterface: True

4
正如您在接受的答案中看到的那样,您互换了的用法类型IsAssignableFrom。就像本杰明(Benjamin)和伊万(Ehouarn)警告一样。
VV5198722 '18年

0

请注意,如果您具有通用接口,IMyInterface<T>则它将始终返回false

  typeof(IMyInterface<>).IsAssignableFrom(typeof(MyType)) /* ALWAYS FALSE */

这也不起作用:

  typeof(MyType).GetInterfaces().Contains(typeof(IMyInterface<>))  /* ALWAYS FALSE */

但是,如果MyType实现IMyInterface<MyType>此功能,则返回true

  typeof(IMyInterface<MyType>).IsAssignableFrom(typeof(MyType))

但是,您可能T在运行时不知道type参数。一个有点古怪的解决方案是:

  typeof(MyType).GetInterfaces()
                .Any(x=>x.Name == typeof(IMyInterface<>).Name)

Jeff的解决方案不太灵活:

  typeof(MyType).GetInterfaces()
         .Any(i => i.IsGenericType 
             && i.GetGenericTypeDefinition() == typeof(IMyInterface<>));

这是一种Type适用于任何情况的扩展方法:

public static class TypeExtensions
{
    public static bool IsImplementing(this Type type, Type someInterface)
    {
        return type.GetInterfaces()
             .Any(i => i == someInterface 
                 || i.IsGenericType 
                    && i.GetGenericTypeDefinition() == someInterface);
    }
}

(请注意,上面使用的是linq,它可能比循环慢。)

然后,您可以执行以下操作:

   typeof(MyType).IsImplementing(IMyInterface<>)
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.