Преобразование в Nullable<T> из строки с помощью отражения
-
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))
как предложил Марк Грэвелл, и для этого потребуется лишь несколько незначительных изменений в коде.