Как определить, является ли тип A неявно конвертируемым в тип B - PullRequest
22 голосов
/ 08 февраля 2010

Учитывая Тип a и Тип b, как я могу во время выполнения определить, существует ли неявное преобразование из a в b?

Если это не имеет смысла, рассмотрите следующий метод:

public PropertyInfo GetCompatibleProperty<T>(object instance, string propertyName)
{
   var property = instance.GetType().GetProperty(propertyName);

   bool isCompatibleProperty = !property.PropertyType.IsAssignableFrom(typeof(T));
   if (!isCompatibleProperty) throw new Exception("OH NOES!!!");

   return property;   
}

А вот код вызова, с которым я хочу работать:

// Since string.Length is an int property, and ints are convertible
// to double, this should work, but it doesn't. :-(
var property = GetCompatibleProperty<double>("someStringHere", "Length");

Ответы [ 3 ]

25 голосов
/ 08 февраля 2010

Обратите внимание, что IsAssignableFrom НЕ решает вашу проблему. Вы должны использовать отражение, как так. Обратите внимание на явную необходимость обработки примитивных типов; эти списки соответствуют §6.1.2 (неявные числовые преобразования) спецификации.

static class TypeExtensions { 
    static Dictionary<Type, List<Type>> dict = new Dictionary<Type, List<Type>>() {
        { typeof(decimal), new List<Type> { typeof(sbyte), typeof(byte), typeof(short), typeof(ushort), typeof(int), typeof(uint), typeof(long), typeof(ulong), typeof(char) } },
        { typeof(double), new List<Type> { typeof(sbyte), typeof(byte), typeof(short), typeof(ushort), typeof(int), typeof(uint), typeof(long), typeof(ulong), typeof(char), typeof(float) } },
        { typeof(float), new List<Type> { typeof(sbyte), typeof(byte), typeof(short), typeof(ushort), typeof(int), typeof(uint), typeof(long), typeof(ulong), typeof(char), typeof(float) } },
        { typeof(ulong), new List<Type> { typeof(byte), typeof(ushort), typeof(uint), typeof(char) } },
        { typeof(long), new List<Type> { typeof(sbyte), typeof(byte), typeof(short), typeof(ushort), typeof(int), typeof(uint), typeof(char) } },
        { typeof(uint), new List<Type> { typeof(byte), typeof(ushort), typeof(char) } },
        { typeof(int), new List<Type> { typeof(sbyte), typeof(byte), typeof(short), typeof(ushort), typeof(char) } },
        { typeof(ushort), new List<Type> { typeof(byte), typeof(char) } },
        { typeof(short), new List<Type> { typeof(byte) } }
    };
    public static bool IsCastableTo(this Type from, Type to) { 
        if (to.IsAssignableFrom(from)) { 
            return true; 
        }
        if (dict.ContainsKey(to) && dict[to].Contains(from)) {
            return true;
        }
        bool castable = from.GetMethods(BindingFlags.Public | BindingFlags.Static) 
                        .Any( 
                            m => m.ReturnType == to &&  
                            (m.Name == "op_Implicit" ||  
                            m.Name == "op_Explicit")
                        ); 
        return castable; 
    } 
} 

Использование:

bool b = typeof(A).IsCastableTo(typeof(B));
5 голосов
/ 15 мая 2014

Принятый ответ на этот вопрос касается многих случаев, но не всех. Например, вот только несколько допустимых приведений / преобразований, которые не обрабатываются правильно:

// explicit
var a = (byte)2;
var b = (decimal?)2M;

// implicit
double? c = (byte)2;
decimal? d = 4L;

Ниже я разместил альтернативную версию этой функции, которая конкретно отвечает на вопрос о приведениях и преобразованиях IMPLICIT. Для получения более подробной информации, комплект тестов, который я использовал для его проверки, и версию EXPLICIT cast, пожалуйста, прочитайте мой пост на тему .

public static bool IsImplicitlyCastableTo(this Type from, Type to)
{
    // from http://www.codeducky.org/10-utilities-c-developers-should-know-part-one/ 
    Throw.IfNull(from, "from");
    Throw.IfNull(to, "to");

    // not strictly necessary, but speeds things up
    if (to.IsAssignableFrom(from))
    {
        return true;
    }

    try
    {
        // overload of GetMethod() from http://www.codeducky.org/10-utilities-c-developers-should-know-part-two/ 
        // that takes Expression<Action>
        ReflectionHelpers.GetMethod(() => AttemptImplicitCast<object, object>())
            .GetGenericMethodDefinition()
            .MakeGenericMethod(from, to)
            .Invoke(null, new object[0]);
        return true;
    }
    catch (TargetInvocationException ex)
    {
        return = !(
            ex.InnerException is RuntimeBinderException
            // if the code runs in an environment where this message is localized, we could attempt a known failure first and base the regex on it's message
            && Regex.IsMatch(ex.InnerException.Message, @"^The best overloaded method match for 'System.Collections.Generic.List<.*>.Add(.*)' has some invalid arguments$")
        );
    }
}

private static void AttemptImplicitCast<TFrom, TTo>()
{
    // based on the IL produced by:
    // dynamic list = new List<TTo>();
    // list.Add(default(TFrom));
    // We can't use the above code because it will mimic a cast in a generic method
    // which doesn't have the same semantics as a cast in a non-generic method

    var list = new List<TTo>(capacity: 1);
    var binder = Microsoft.CSharp.RuntimeBinder.Binder.InvokeMember(
        flags: CSharpBinderFlags.ResultDiscarded, 
        name: "Add", 
        typeArguments: null, 
        context: typeof(TypeHelpers), // the current type
        argumentInfo: new[] 
        { 
            CSharpArgumentInfo.Create(flags: CSharpArgumentInfoFlags.None, name: null), 
            CSharpArgumentInfo.Create(
                flags: CSharpArgumentInfoFlags.UseCompileTimeType, 
                name: null
            ),
        }
    );
    var callSite = CallSite<Action<CallSite, object, TFrom>>.Create(binder);
    callSite.Target.Invoke(callSite, list, default(TFrom));
}
5 голосов
/ 08 февраля 2010

Неявные преобразования, которые необходимо учитывать:

  • Идентичность
  • sbyte to short, int, long, float, double или decimal
  • байт в короткую, ushort, int, uint, long, ulong, float, double или decimal
  • short to int, long, float, double или decimal
  • короткая к int, uint, long, ulong, float, double или decimal
  • int в long, float, double или decimal
  • От Uint до Long, Ulong, Float, Double или Decimal
  • длинные числа с плавающей запятой, двойные или десятичные числа
  • ulong to float, double или decimal
  • char в ushort, int, uint, long, ulong, float, double или decimal
  • с плавающей запятой, чтобы удвоить
  • Преобразование типа Nullable
  • Тип ссылки на объект
  • Производный класс в базовый класс
  • Класс для реализованного интерфейса
  • Интерфейс к базовому интерфейсу
  • Массив в массив, если у массивов одинаковое количество измерений, существует неявное преобразование из типа исходного элемента в тип элемента назначения, а тип исходного элемента и тип элемента назначения являются ссылочными типами
  • Тип массива в System.Array
  • Тип массива в IList <> и его базовые интерфейсы
  • Тип делегата для System.Delegate
  • Бокс конверсия
  • Тип перечисления для System.Enum
  • Определяемое пользователем преобразование (op_implicit)

Полагаю, вы ищете последнее. Вам нужно написать что-то похожее на компилятор, чтобы охватить все из них. Следует отметить, что System.Linq.Expressions.Expression не пытался выполнить этот подвиг.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...