从'System.Int32'到'System.Nullable`1 [[System.Int32,mscorlib]]的无效转换


81
Type t = typeof(int?); //will get this dynamically
object val = 5; //will get this dynamically
object nVal = Convert.ChangeType(val, t);//getting exception here

我在上面的代码中收到InvalidCastException。对于上述内容,我可以简单地编写int? nVal = val,但是以上代码是动态执行的。

我正在将一个值(非空类型,如int,float等)包装在一个对象(在此为val)中,并且我必须通过将其强制转换为另一个类型(可以为空的版本,也可以为不能为空的版本)来将其保存到另一个对象的)。什么时候

从'System.Int32'到'System.Nullable`1 [[System.Int32,mscorlib,Version = 4.0.0.0,Culture = neutral,PublicKeyToken = b77a5c561934e089]]的无效转换。

一个int,应该是可转换/类型可转换的nullable int,这是什么问题?


我猜可能Nullable<T>是因为IConvertible
Coz

2
这是非常基本的。Nullable是特殊的,当您将其放入对象中时,它要么变为null要么变为值类型的装箱值。所以要求一个整数?存储在对象中只是没有意义。只是要求整数。
汉斯·帕桑

Answers:


140

您必须使用Nullable.GetUnderlyingType来获取的基础类型Nullable

这是我用来克服ChangeTypefor限制的方法Nullable

public static T ChangeType<T>(object value) 
{
   var t = typeof(T);

   if (t.IsGenericType && t.GetGenericTypeDefinition().Equals(typeof(Nullable<>))) 
   {
       if (value == null) 
       { 
           return default(T); 
       }

       t = Nullable.GetUnderlyingType(t);
   }

   return (T)Convert.ChangeType(value, t);
}

非通用方法:

public static object ChangeType(object value, Type conversion) 
{
   var t = conversion;

   if (t.IsGenericType && t.GetGenericTypeDefinition().Equals(typeof(Nullable<>))) 
   {
       if (value == null) 
       { 
           return null; 
       }

       t = Nullable.GetUnderlyingType(t);
   }

   return Convert.ChangeType(value, t);
}

为了使用您的方法,我需要做一些类似的事情:object nVal = ChangeType<int?>(val),这里我需要告诉该方法有关通用参数(T)的信息,但是我可以使用t(或typeof(dataType))。在我的场景中,我将如何调用您的ChangeType方法?
2013年

1
添加了非通用版本。看看是否有帮助。
gzaxx

在出现编译错误default(conversion),似乎是类似的问题。
Brij

小心@gzaxx与return null不一样default(T)。如果要处理结构,则它们是完全不同的东西。
Alex

非通用版本将框类型和基本类型(因为它接受并返回对象),所以返回null是有效的。调用函数的任何人都必须自己处理。
gzaxx

9

以上我可以简单地写int吗?nVal = val

实际上,您也不能这样做。没有从object到的隐式转换Nullable<int>。但是,从隐式转换intNullable<int>,所以你可以这样写:

int? unVal = (int)val;

您可以使用Nullable.GetUnderlyingType方法。

返回指定可空类型的基础类型参数

泛型类型定义是一种类型声明,例如Nullable,它包含一个类型参数列表,并且该类型参数列表声明一个或多个类型参数。封闭的泛型类型是类型声明,其中为类型参数指定了特定类型。

Type t = typeof(int?); //will get this dynamically
Type u = Nullable.GetUnderlyingType(t);
object val = 5; //will get this dynamically
object nVal = Convert.ChangeType(val, u);// nVal will be 5

这是一个DEMO


2

我想我应该解释一下为什么该功能不起作用:

1-引发异常的行如下:

throw new InvalidCastException(Environment.GetResourceString("InvalidCast_FromTo", new object[]
  {
    value.GetType().FullName, 
    targetType.FullName
    }));

实际上,该函数在数组Convert.ConvertTypes中搜索之后,便会查看目标程序是否为Enum,并且在未找到任何内容的情况下,它将引发上面的异常。

2-Convert.ConvertTypes初始化为:

Convert.ConvertTypes = new RuntimeType[]
   {
      (RuntimeType)typeof(Empty), 
      (RuntimeType)typeof(object), 
      (RuntimeType)typeof(DBNull), 
      (RuntimeType)typeof(bool), 
      (RuntimeType)typeof(char), 
      (RuntimeType)typeof(sbyte), 
      (RuntimeType)typeof(byte), 
      (RuntimeType)typeof(short), 
      (RuntimeType)typeof(ushort), 
      (RuntimeType)typeof(int), 
      (RuntimeType)typeof(uint), 
      (RuntimeType)typeof(long), 
      (RuntimeType)typeof(ulong), 
      (RuntimeType)typeof(float), 
      (RuntimeType)typeof(double), 
      (RuntimeType)typeof(decimal), 
      (RuntimeType)typeof(DateTime), 
      (RuntimeType)typeof(object), 
      (RuntimeType)typeof(string)
   };

因此,由于int?不在ConvertTypes数组中,并且不在枚举中,因此将引发异常。

因此要恢复,要使功能Convert.ChnageType起作用,您需要:

  1. 要转换的对象是IConvertible

  2. 目标类型是ConvertTypes内,而不是Empty也不DBNull(有关于这两个显式的测试与抛出异常)

这是因为int(和所有其他默认类型)Convert.DefaultToType用作IConvertibale.ToType implementation. and here is the code of theDefaultToTypeextracted使用ILSpy

internal static object DefaultToType(IConvertible value, Type targetType, IFormatProvider provider)
{
    if (targetType == null)
    {
        throw new ArgumentNullException("targetType");
    }
    RuntimeType left = targetType as RuntimeType;
    if (left != null)
    {
        if (value.GetType() == targetType)
        {
            return value;
        }
        if (left == Convert.ConvertTypes[3])
        {
            return value.ToBoolean(provider);
        }
        if (left == Convert.ConvertTypes[4])
        {
            return value.ToChar(provider);
        }
        if (left == Convert.ConvertTypes[5])
        {
            return value.ToSByte(provider);
        }
        if (left == Convert.ConvertTypes[6])
        {
            return value.ToByte(provider);
        }
        if (left == Convert.ConvertTypes[7])
        {
            return value.ToInt16(provider);
        }
        if (left == Convert.ConvertTypes[8])
        {
            return value.ToUInt16(provider);
        }
        if (left == Convert.ConvertTypes[9])
        {
            return value.ToInt32(provider);
        }
        if (left == Convert.ConvertTypes[10])
        {
            return value.ToUInt32(provider);
        }
        if (left == Convert.ConvertTypes[11])
        {
            return value.ToInt64(provider);
        }
        if (left == Convert.ConvertTypes[12])
        {
            return value.ToUInt64(provider);
        }
        if (left == Convert.ConvertTypes[13])
        {
            return value.ToSingle(provider);
        }
        if (left == Convert.ConvertTypes[14])
        {
            return value.ToDouble(provider);
        }
        if (left == Convert.ConvertTypes[15])
        {
            return value.ToDecimal(provider);
        }
        if (left == Convert.ConvertTypes[16])
        {
            return value.ToDateTime(provider);
        }
        if (left == Convert.ConvertTypes[18])
        {
            return value.ToString(provider);
        }
        if (left == Convert.ConvertTypes[1])
        {
            return value;
        }
        if (left == Convert.EnumType)
        {
            return (Enum)value;
        }
        if (left == Convert.ConvertTypes[2])
        {
            throw new InvalidCastException(Environment.GetResourceString("InvalidCast_DBNull"));
        }
        if (left == Convert.ConvertTypes[0])
        {
            throw new InvalidCastException(Environment.GetResourceString("InvalidCast_Empty"));
        }
    }
    throw new InvalidCastException(Environment.GetResourceString("InvalidCast_FromTo", new object[]
    {
        value.GetType().FullName, 
        targetType.FullName
    }));
}

另一方面,强制转换由Nullable类本身实现,其定义为:

public static implicit operator T?(T value)
{
    return new T?(value);
}
public static explicit operator T(T? value)
{
    return value.Value;
}
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.