Преобразование в Nullable<T> из строки с помощью отражения

StackOverflow https://stackoverflow.com/questions/894640

  •  23-08-2019
  •  | 
  •  

Вопрос

Как я могу преобразовать в Nullable из строки, используя отражение?

У меня есть следующий код для преобразования почти любого типа значения с учетом почти любого значения.Выше есть довольно много кода для использования IsAssignableFrom и т. д.так что это крайняя мера.

MethodInfo parse = t.GetMethod("Parse", new Type[] { typeof(string) });

if (parse != null)
{
    object parsed = parse.Invoke(null, new object[] { value.ToString() });
    return (T)parsed;
}
else
{
    throw new InvalidOperationException("The value you specified is not a valid " + typeof(T).ToString());
}

Проблема возникает, когда я хочу преобразовать тип, допускающий значение NULL, например long?.

Очевидно, длинный?у класса нет метода синтаксического анализа.

Как извлечь метод синтаксического анализа из нулевых значений шаблон тип?

РЕДАКТИРОВАТЬ:

Вот небольшой набор тестов, которые я пытаюсь пройти:

[Test]
public void ConverterTNullable()
{
    Assert.That((int?)1, Is.EqualTo(Converter<int?>.Convert(1)));
    Assert.That((int?)2, Is.EqualTo(Converter<int?>.Convert(2.0d)));
    Assert.That(3, Is.EqualTo(Converter<long>.Convert(3)));

    Assert.That((object)null, Is.EqualTo(Converter<long?>.Convert("")));
    Assert.That((object)null, Is.EqualTo(Converter<long?>.Convert(null)));
    Assert.That((object)null, Is.EqualTo(Converter<long?>.Convert(DBNull.Value)));

    Assert.That((long)1, Is.EqualTo(Converter<long?>.Convert("1")));
    Assert.That((long)2, Is.EqualTo(Converter<long?>.Convert(2.0)));
    Assert.That((long?)3, Is.EqualTo(Converter<long>.Convert(3)));
}

И вся функция:

/// <summary>
/// Converts any compatible object to an instance of T.
/// </summary>
/// <param name="value">The value to convert.</param>
/// <returns>The converted value.</returns>
public static T Convert(object value)
{
    if (value is T)
    {
        return (T)value;
    }

    Type t = typeof(T);

    if (t == typeof(string))
    {
        if (value is DBNull || value == null)
        {
            return (T)(object)null;
        }
        else
        {
            return (T)(object)(value.ToString());
        }
    }
    else
    {
        if (value is DBNull || value == null)
        {
            return default(T);
        }

        if (value is string && string.IsNullOrEmpty((string)value))
        {
            return default(T);
        }

        try
        {
            return (T)value;
        }
        catch (InvalidCastException)
        {
        }

        if (Nullable.GetUnderlyingType(t) != null)
        {
            t = Nullable.GetUnderlyingType(t);
        }

        MethodInfo parse = t.GetMethod("Parse", new Type[] { typeof(string) });

        if (parse != null)
        {
            object parsed = parse.Invoke(null, new object[] { value.ToString() });
            return (T)parsed;
        }
        else
        {
            throw new InvalidOperationException("The value you specified is not a valid " + typeof(T).ToString());
        }
    }
}
Это было полезно?

Решение 3

Я добавил немного:

if (Nullable.GetUnderlyingType(t) != null)
{
    t = Nullable.GetUnderlyingType(t);
}

MethodInfo parse = t.GetMethod("Parse", new Type[] { typeof(string) });

Всем спасибо!

Другие советы

Я бы, наверное, использовал TypeConverter в этом случае, и Nullable.GetUnderlyingType();пример на подходе...

    static void Main()
    {
        long? val1 = Parse<long?>("123");
        long? val2 = Parse<long?>(null);
    }

    static T Parse<T>(string value)
    {
        return (T) TypeDescriptor.GetConverter(typeof(T))
            .ConvertFrom(value);
    }

Это должно помочь в работе.(Для простоты код встроен в метод расширения, хотя это может быть не то, что вам нужно.)

public static T? ParseToNullable<T>(this string value) where T : struct
{
    var parseMethod = typeof(T).GetMethod("Parse", new Type[] { typeof(string) });
    if (parseMethod == null)
        return new Nullable<T>();

    try
    {
        var value = parseMethod.Invoke(null, new object[] { value.ToString() });
        return new Nullable<T>((T)value);
    }
    catch
    {
        return new Nullable<T>();
    }
}

Теперь, если вам нужен параметр универсального типа сам чтобы быть типом, допускающим значение NULL, а не базовым (я не вижу в этом преимущества), вы можете использовать:

Nullable.GetUnderlyingType(typeof(T))

как предложил Марк Грэвелл, и для этого потребуется лишь несколько незначительных изменений в коде.

Лицензировано под: CC-BY-SA с атрибуция
Не связан с StackOverflow
scroll top