【发布时间】:2011-03-30 23:18:42
【问题描述】:
我想根据泛型返回类型将字符串转换为泛型类型,如int 或date 或long。
基本上是一个类似Parse<T>(String) 的函数,它返回T 类型的项目。
例如,如果传递了一个 int,则函数应该在内部执行 int.parse。
【问题讨论】:
标签: c# string parsing type-conversion
我想根据泛型返回类型将字符串转换为泛型类型,如int 或date 或long。
基本上是一个类似Parse<T>(String) 的函数,它返回T 类型的项目。
例如,如果传递了一个 int,则函数应该在内部执行 int.parse。
【问题讨论】:
标签: c# string parsing type-conversion
根据您的示例,您可以这样做:
int i = (int)Convert.ChangeType("123", typeof(int));
DateTime dt = (DateTime)Convert.ChangeType("2009/12/12", typeof(DateTime));
为了满足您的“通用返回类型”要求,您可以编写自己的扩展方法:
public static T ChangeType<T>(this object obj)
{
return (T)Convert.ChangeType(obj, typeof(T));
}
这将允许您这样做:
int i = "123".ChangeType<int>();
【讨论】:
IConvertable不应该也约束T,即T ChangeType<T>(this object obj) where T : IConvertable吗?
obj 必须是 IConvertible,但无法在编译时指定。
看来我来不及回答这个问题了。但这是我的实现:
基本上,我已经为 Object 类创建了一个扩展方法。它处理所有类型,即可为空、类和结构。
public static T ConvertTo<T>(this object value)
{
T returnValue;
if (value is T variable)
returnValue = variable;
else
try
{
//Handling Nullable types i.e, int?, double?, bool? .. etc
if (Nullable.GetUnderlyingType(typeof(T)) != null)
{
TypeConverter conv = TypeDescriptor.GetConverter(typeof(T));
returnValue = (T) conv.ConvertFrom(value);
}
else
{
returnValue = (T) Convert.ChangeType(value, typeof(T));
}
}
catch (Exception)
{
returnValue = default(T);
}
return returnValue;
}
【讨论】:
TypeDescriptor 可空类型和Convert.ChangeType 不可空类型是否有特定原因?整个try 块只能减少到TypeConverter 2 行代码,它适用于可空和不可空。
Pranay's answer 的更简洁版本
public static T ConvertTo<T>(this object value)
{
if (value is T variable) return variable;
try
{
//Handling Nullable types i.e, int?, double?, bool? .. etc
if (Nullable.GetUnderlyingType(typeof(T)) != null)
{
return (T)TypeDescriptor.GetConverter(typeof(T)).ConvertFrom(value);
}
return (T)Convert.ChangeType(value, typeof(T));
}
catch (Exception)
{
return default(T);
}
}
【讨论】:
【讨论】:
.NET 中有几个约定可以将一种类型的对象转换为另一种类型。
但是这些方法比典型的T.Parse(string) 慢得多,会导致装箱,并且每次要转换单个值时都涉及大量分配。
对于ValueString,我选择使用反射找到一个合适的类型的静态解析方法,构建一个调用它的 lambda 表达式并缓存已编译的委托以供将来使用(参见this answer 的示例)。
如果类型没有合适的解析方法,它也会回退到我上面提到的方法(参见自述文件中的performance section)。
var v = new ValueString("15"); // struct
var i = v.As<int>(); // Calls int.Parse.
【讨论】: