我如何检查类型是否是子类型或对象的类型?

要在c#中检查一个类型是否是另一个类型的子类,很简单:

typeof (SubClass).IsSubclassOf(typeof (BaseClass)); // returns true

然而,这将失败:

typeof (BaseClass).IsSubclassOf(typeof (BaseClass)); // returns false

是否有任何方法可以在不使用OR操作符或扩展方法的情况下检查类型是子类还是基类本身?

229275 次浏览
typeof(BaseClass).IsAssignableFrom(unknownType);

你应该尝试使用类型。IsAssignableFrom代替。

很明显,没有。

以下是选项:

类型。IsSubclassOf

正如你已经发现的,如果两种类型相同,这将不起作用,下面是一个示例LINQPad程序,演示:

void Main()
{
typeof(Derived).IsSubclassOf(typeof(Base)).Dump();
typeof(Base).IsSubclassOf(typeof(Base)).Dump();
}


public class Base { }
public class Derived : Base { }

输出:

True
False

这表明DerivedBase的子类,但Base(显然)不是它自己的子类。

类型。IsAssignableFrom

这可以回答你的问题,但也会给你假阳性。正如Eric Lippert在评论中指出的,虽然该方法确实会为上述两个问题返回True,但它也会为这些问题返回True,这可能是你不想要的:

void Main()
{
typeof(Base).IsAssignableFrom(typeof(Derived)).Dump();
typeof(Base).IsAssignableFrom(typeof(Base)).Dump();
typeof(int[]).IsAssignableFrom(typeof(uint[])).Dump();
}


public class Base { }
public class Derived : Base { }

这里你会得到以下输出:

True
True
True

如果方法只有回答了所问的问题,那么最后一个True将表明uint[]继承自int[],或者它们是同一类型,但显然不是这样。

所以IsAssignableFrom也不完全正确。

isas

在你的问题上下文中,isas的“问题”是,它们将要求你对对象进行操作,并直接在代码中编写其中一种类型,而不能与Type对象一起工作。

换句话说,这将不会编译:

SubClass is BaseClass
^--+---^
|
+-- need object reference here

这个也不会:

typeof(SubClass) is typeof(BaseClass)
^-------+-------^
|
+-- need type name here, not Type object

这个也不会:

typeof(SubClass) is BaseClass
^------+-------^
|
+-- this returns a Type object, And "System.Type" does not
inherit from BaseClass

结论

虽然上述方法可能适合你的需求,但对你的问题(在我看来)唯一正确的答案是你需要额外检查:

typeof(Derived).IsSubclassOf(typeof(Base)) || typeof(Derived) == typeof(Base);

当然,这在方法中更有意义:

public bool IsSameOrSubclass(Type potentialBase, Type potentialDescendant)
{
return potentialDescendant.IsSubclassOf(potentialBase)
|| potentialDescendant == potentialBase;
}

我张贴这个答案,希望有人能和我分享这是否是一个坏主意,以及为什么这是一个坏主意。在我的应用程序中,我有一个类型的属性,我想检查,以确保它是typeof(a)或typeof(B),其中B是从a派生的任何类,所以我的代码:

public class A
{
}


public class B : A
{
}


public class MyClass
{
private Type _helperType;
public Type HelperType
{
get { return _helperType; }
set
{
var testInstance = (A)Activator.CreateInstance(value);
if (testInstance==null)
throw new InvalidCastException("HelperType must be derived from A");
_helperType = value;
}
}
}

我觉得我可能有点天真,所以任何反馈都会受到欢迎。

如果你试图在Xamarin Forms PCL项目中这样做,上面的解决方案使用IsAssignableFrom会给出一个错误:

错误:'Type'不包含'IsAssignableFrom'的定义 没有扩展方法'IsAssignableFrom'接受的第一个参数 类型'类型'可以找到(你是否缺少一个使用指令或一个 装配参考?)< / p >

,因为IsAssignableFrom请求TypeInfo对象。 你可以使用System.Reflection中的GetTypeInfo()方法:

typeof(BaseClass).GetTypeInfo().IsAssignableFrom(typeof(unknownType).GetTypeInfo())