当返回类型是通用接口时,如何将对象强制转换为泛型返回类型?

时间:2015-12-30 22:25:55

标签: c# generics iconvertible

我有一个非常丑陋的方法,它应该从数据库中提取各种数据,并使用它构造一个请求类型的对象。目标是将各种丑陋归结到一个地方,以便从包含此方法的类派生的类可以相对干净。

即使该类型可以为空,我也不会有什么问题,即使该类型可以为空:

protected T getValue<T>(ParameterObject myParameterObject)
{
    var typeOfT = typeof (T);

    if (typeOfT == typeof(bool))
    {
        string val = this.getValue(myParameterObject);

        if (val == null)
            return default(T);

        bool rVal;
        if (!Boolean.TryParse(val, out rVal))
            return default(T);

        return changeType<T>(rVal);
    }

    if (typeOfT == typeof (double?))
    {
        string val = this.getValue(myParameterObject);

        if (val == null)
            return default(T);

        double rVal;
        if (!Double.TryParse(val, out rVal))
            return default(T);

        return changeType<T>(rVal);
    }

    [...]
}

public static T changeType<T>(object value)
{
    var t = typeof(T);
    if (!t.IsGenericType || t.GetGenericTypeDefinition() != typeof(Nullable<>))
        return (T)Convert.ChangeType(value, t);

    if (value == null)
        return default(T);

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

不幸的是,我的情况是我想要的返回类型是IList,我想返回一个List:

protected T getValue<T>(ParameterObject myParameterObject)
{
    var typeOfT = typeof (T);

    if (typeOfT.IsGenericType)
    {
        if (typeOfT.GetGenericTypeDefinition() == typeof (IList<>))
        {
            var genericArgumentType = typeOfT.GenericTypeArguments.FirstOrDefault();
            var baseType = typeof (List<>);
            var genericType = baseType.MakeGenericType(new[] {genericArgumentType});

            var rVal = Activator.CreateInstance(genericType);

            [code to populate the list] 

            return changeType<T>(rVal);
        }

    [...]
}

我无法返回一个rVal,因为它是一个对象,而不是T的实例。但是我的changeType()函数失败了,并出现错误:

  

System.InvalidCastException:Object必须实现IConvertible。

所以,我有一个泛型方法,返回类型是IList,我有一个List的对象,但我对它的引用是Object类型。如何投射它以便我可以从函数中返回它?

1 个答案:

答案 0 :(得分:1)

您的return (T)Convert.ChangeType(value, t);行失败,因为您的value对象未实现IConvertible

来自MSDN docs

  

[...]为了使转换成功,值必须实现IConvertible接口,因为该方法只是调用适当的IConvertible方法。该方法要求支持将值转换为conversionType。

要解决这个问题,我不确定一般方法,但对于value类型为IEnumerableIEnumerable<>类型的特定情况,您可以尝试施放然后执行value.Select(x => changeType<ListGenT>(x)) linq查询。通过递归使用changeType方法,这也应该处理嵌套的泛型类型。