'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上記の場合、単純に書くことができますが、上記のコードは動的に実行されます。

オブジェクト(ここではval)にラップされた値(int、floatなどのnull許容型ではない)を取得しています。別の型(null許容バージョンにすることもできないこともあります)にキャストして、別のオブジェクトに保存する必要があります。それの)。いつ

'System.Int32'から 'System.Nullable`1 [[System.Int32、mscorlib、Version = 4.0.0.0、Culture = neutral、PublicKeyToken = b77a5c561934e089]]'へのキャストが無効です。

int、にコンバーチブル/タイプキャスト可能である必要がありますnullable intが、ここでの問題は何ですか?


cozNullable<T>は実装されていないのではないかと思いますIConvertible
V4Vendetta 2013

2
これはかなり基本的なことです。Nullableは特別です。オブジェクトに入れると、nullになるか、値型のボックス化された値になります。それで、intを求めますか?オブジェクトに格納しても意味がありません。intを要求するだけです。
ハンスパッサント2013

回答:


143

Nullable.GetUnderlyingType基になるタイプを取得するには、を使用する必要がありNullableます。

これは私がの制限を克服するために使用する方法ChangeTypeです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メソッドをどのように呼び出しますか?
Brij 2013

1
非汎用バージョンを追加しました。それが役立つかどうかを確認してください。
gzaxx 2013

でコンパイルエラーが発生するのdefault(conversion)も同様の問題のようです。
Brij 2013

return nullと同じではないので注意してください@gzaxx default(T)。構造体を扱っている場合、それらは完全に異なるものです。
アレックス

非汎用バージョンは、構造体型とプリミティブ型をボックス化するため(オブジェクトを取得して返すため)、nullを返すことは有効です。関数を呼び出す人は誰でもこれを自分で処理する必要があります。
gzaxx 2016年

9

上記の場合、単純にintと書くことができますか?nVal = val

実際、それもできません。からobjectへの暗黙の変換はありませんNullable<int>。しかし、そこにあるから暗黙的な変換intNullable<int>あなたがこれを書くことができるようには:

int? unVal = (int)val;

Nullable.GetUnderlyingTypeメソッドを使用できます。

指定されたnull許容型の基になる型引数を返します。

ジェネリック型定義は、型パラメーターリストを含むNullableなどの型宣言であり、型パラメーターリストは1つ以上の型パラメーターを宣言します。クローズドジェネリック型は、特定の型が型パラメーターに指定されている型宣言です。

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で関数を検索すると、ターゲットが列挙型であるかどうかが確認され、何も見つからない場合は上記の例外がスローされます。

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内にあり、norではありませEmptyDBNull(throw例外を除いてこれら2つに明示的なテストがあります)

ので、この動作は、int(他のすべてのデフォルトのタイプ)は使用していますConvert.DefaultToTypeIConvertibale.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;
}
弊社のサイトを使用することにより、あなたは弊社のクッキーポリシーおよびプライバシーポリシーを読み、理解したものとみなされます。
Licensed under cc by-sa 3.0 with attribution required.