通用方法,取消装箱可为空的枚举

har*_*ldr 5 c# generics enums unboxing nullable

我做了以下扩展方法......

public static class ObjectExtensions
{
    public static T As<T>(this object pObject, T pDefaultValue)
    {
        if (pObject == null || pObject == DBNull.Value)
            return pDefaultValue;
        return (T) pObject;
    }
}
Run Code Online (Sandbox Code Playgroud)

...我用于例如读取这样的数据:

string field = datareader["column"].As("default value when null")
Run Code Online (Sandbox Code Playgroud)

但是当我想从装箱值转换为可为空的枚举时,它不起作用。我能想到的最好的就是这个(无效的杂乱 WIP 代码):

public static class ObjectExtensions
{
    public static T As<T>(this object pObject, T pDefaultValue)
    {
        if (pObject == null || pObject == DBNull.Value)
            return pDefaultValue;

        var lType = typeof (T);

        if (!IsNullableEnum(lType))
            return (T) pObject;

        var lEnumType = Nullable.GetUnderlyingType(lType);
        var lEnumPrimitiveType = lEnumType.GetEnumUnderlyingType();

        if (lEnumPrimitiveType == typeof(int))
        {
            var lObject = (int?) pObject;
            return (T) Convert.ChangeType(lObject, lType);
        }

        throw new InvalidCastException();
    }

    private static bool IsNullableEnum(Type pType)
    {
        Type lUnderlyingType = Nullable.GetUnderlyingType(pType);
        return (lUnderlyingType != null) && lUnderlyingType.IsEnum;
    }
}
Run Code Online (Sandbox Code Playgroud)

用法:

public enum SomeEnum {Value1, Value2};
object value = 1;
var result = value.As<SomeEnum?>();
Run Code Online (Sandbox Code Playgroud)

当前错误是 InvalidCastException 当它尝试将 Int32 转换为可空枚举时。我想这没问题,但我不知道我还能怎么做?我试图创建一个可为空枚举 T 的实例并为其分配一个值,但我一直在思考如何才能做到这一点。

任何人的想法或更好的方法来解决这个问题?甚至有可能以通用的方式解决这个问题吗?我已经做了很多搜索,但我没有发现任何有用的东西。

Han*_*ant 2

您可以通过调用所需的可为空类型的构造函数来完成此操作。像这样:

            Type t = typeof(Nullable<>).MakeGenericType(lEnumType);
            var ctor = t.GetConstructor(new Type[] { lEnumType });
            return (T)ctor.Invoke(new object[] { pObject });
Run Code Online (Sandbox Code Playgroud)