我有一个非常丑陋的方法,它应该从数据库中提取各种数据,并使用它构造一个请求类型的对象。目标是将各种丑陋归结到一个地方,以便从包含此方法的类派生的类可以相对干净。
即使该类型可以为空,我也不会有什么问题,即使该类型可以为空:
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类型。如何投射它以便我可以从函数中返回它?
答案 0 :(得分:1)
您的return (T)Convert.ChangeType(value, t);
行失败,因为您的value
对象未实现IConvertible
。
[...]为了使转换成功,值必须实现IConvertible接口,因为该方法只是调用适当的IConvertible方法。该方法要求支持将值转换为conversionType。
要解决这个问题,我不确定一般方法,但对于value
类型为IEnumerable
或IEnumerable<>
类型的特定情况,您可以尝试施放然后执行value.Select(x => changeType<ListGenT>(x))
linq查询。通过递归使用changeType
方法,这也应该处理嵌套的泛型类型。