Check if type TA can cast to type TB at runtime? (

2019-05-04 07:21发布

So I know that this works:

class A
{
}
class B : A
{       
}

[Test]
public void CanCast()
{
    Assert.That(typeof(A).IsAssignableFrom(typeof(B)));
    Assert.That(!typeof(B).IsAssignableFrom(typeof(A)));
}

However, let's say those two types were Int32 and Int64.

At runtime, I can cast an Int32 value to an Int64 variable, but not the other way around. How can I check this kind of casting-compatibility at runtime? (IsAssignableFrom doesn't work for this, it always gives false for Int32 and Int64)

EDIT: I cannot simply try to cast, because I don't have any value of those types, I'm asking the hypothetical scenario of having two types A and B, not having two values a and b.

标签: c# .net cil
2条回答
smile是对你的礼貌
2楼-- · 2019-05-04 07:52

One (less-than-elegant) approach is to simply try it - wrap your attempt in a try/catch and Assert false if you catch an exception.

叼着烟拽天下
3楼-- · 2019-05-04 08:09

For non-primitive types, you can reflect and inspect whether there is an op_Implicit method on either type supporting the conversion. IL doesn't actually support true operator overloading, so it's purely a convention system for C# to recognize the operator overloads. The method will also be marked IsSpecialName if it was created from an operator overload definition in C#.

For primitive types (like Int32 and Int64), the simplest option is to hardcode the various cases, as conversion is via a primitive IL opcode rather than via a method. There are only a handful of primitive types, though, so it wouldn't be difficult to create a method with the check for all possibilities for each primitive type.

One side note since your example mentioned primitive value types specifically, note that the existence of an implicit 'conversion' (in C#) does not mean that all 'casts' will work. The C# cast operation (T)x can also mean 'unbox the value in x to type T'. If x contains a boxed Int32 and you attempt to (Int64)x, this will fail at runtime, even though you can implicitly 'convert' an Int32 to an Int64. See Eric Lippert for more information as to why unboxing works this way.

查看更多
登录 后发表回答