Как преобразовать строку в обнуляемый тип, который определяется во время выполнения? - PullRequest
16 голосов
/ 24 февраля 2012

У меня есть код ниже, и мне нужно преобразовать строку в тип, который также указан из String:

 Type t = Type.GetType("System.Nullable`1[[System.DateTime, mscorlib, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089]]");

            object d = Convert.ChangeType("2012-02-23 10:00:00", t);

Я получаю сообщение об ошибке ниже:

Invalid cast from 'System.String' to 'System.Nullable`1[[System.DateTime, mscorlib, Version=4.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089]]'.

Как это было бы возможно?

Я знаю, что одним уродливым способом было бы проверить, можно ли обнулять тип, используя if:

    Type possiblyNullableType = Type.GetType("System.Nullable`1[[System.DateTime, mscorlib, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089]]");

    var underlyingType = Nullable.GetUnderlyingType(possiblyNullableType);

    Object result;

    // if it's null, it means it wasn't nullable
    if (underlyingType != null)
    {
        result = Convert.ChangeType("2012-02-23 10:00:00", underlyingType);
    }

Есть ли лучший способ?

Спасибо,

Ответы [ 4 ]

30 голосов
/ 24 февраля 2012

Есть две проблемы.

Во-первых, Convert.ChangeType просто не поддерживает обнуляемые типы.

Во-вторых, даже если бы это было так, поместив результат (назначив его на object), вы уже преобразовали бы его в DateTime.

Вы можете использовать в особых случаях следующие типы:

string s = "2012-02-23 10:00:00";
Type t = Type.GetType("System.Nullable`1[[System.DateTime, mscorlib, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089]]");
object d;

if (t.IsGenericType && t.GetGenericTypeDefinition() == typeof(Nullable<>))
{
    if (String.IsNullOrEmpty(s))
        d = null;
    else
        d = Convert.ChangeType(s, t.GetGenericArguments()[0]);
}
else
{
    d = Convert.ChangeType(s, t);
}
10 голосов
/ 24 февраля 2012

Я написал приведенный ниже универсальный вспомогательный метод, который работает в большинстве сценариев (не тестировался с универсальными типами):

static void Main(string[] args)
{
    Object result =
        ConvertValue(
            "System.Nullable`1[[System.DateTime, mscorlib, Version=2.0.0.0, Culture=neutral, PublicKeyToken=b77a5c561934e089]]",
            "2012-02-23 10:00:00");
}

public static Object ConvertValue(string typeInString, string value)
{
    Type originalType = Type.GetType(typeInString);

    var underlyingType = Nullable.GetUnderlyingType(originalType);

    // if underlyingType has null value, it means the original type wasn't nullable
    object instance = Convert.ChangeType(value, underlyingType ?? originalType);

    return instance;
}
2 голосов
/ 31 июля 2016
public static T GetValue<T>(string Literal, T DefaultValue)
    {
        if (Literal == null || Literal == "" || Literal == string.Empty) return DefaultValue;
        IConvertible obj = Literal;
        Type t = typeof(T);
        Type u = Nullable.GetUnderlyingType(t);

        if (u != null)
        {
            return (obj == null) ? DefaultValue : (T)Convert.ChangeType(obj, u);
        }
        else
        {
            return (T)Convert.ChangeType(obj, t);
        }
    }
1 голос
/ 24 февраля 2012

Как то так?Если вам действительно не нужно делать это динамически.

if (string.IsNullOrEmpty(input))
{
   return new DateTime?();
}
else
{
   return new DateTime?(DateTime.Parse(input));
}

Возможно, вы могли бы проверить, является ли ваш тип одним из 'обнуляемых' типов, и тогда, возможно, вы могли бы найти что-то полезное здесь:

Преобразование строки в обнуляемый тип (int, double и т.д ...)

...