c#中Convert.ChangeTypede意义?泛型方法返回值类型转换

**

应用场景一:

**

如果你知道当前类型应当为string或其他已知类型的化的话,转换没有意义,ChangeType往往用在不知道当前类型应当是什么的情况下,比如有个泛型方法

要求从double类型转换为指定的T类型,因为显式转为T肯定是不行的,因为此时T可能时任何一种对象,只能ChangeType来进行转换。

重点:此方法要求支持将 value 转换为 conversionType。

例如:我想转成int

       int xx = GetValue<int>("1");

        private static T GetValue<T>(object obj)
        {
            return (T)Convert.ChangeType(obj, typeof(T));
        }

ChangeType 是将 value 指定的对象转换为 conversionType 的通用转换方法。 value 参数可以是任何类型的对象,conversionType 也可以是表示任何基类型或自定义类型的 Type 对象。 要使转换成功,value 必须实现 IConvertible 接口,因为此方法只是包装对相应 IConvertible 方法的调用。 此方法要求支持将 value 转换为 conversionType。

应用场景二:

反射操作时经常遇到类型转换操作,但系统的System.Convert.ChangeType不支持枚举ENUM和可空类型Nullable转换。
使用强化版System.Convert.ChangeType前:

enum MyEnum
{
Test1=0,
Test2=1
}
class model
{
public DateTime? StartDate{get;set;}
public MyEnum MyType{get;set;}
}
Type t = typeof (model);
PropertyInfo[] props = t.GetProperties();
var m=new model();
var p1=props.FirstOrDefault(p=>p.Name.Equals("StartDate"));
p1.SetValue(m, Convert.ChangeType(DateTime.Now, p1.PropertyType), null);

这里为可空类型Nullable偿试赋值会产生异常。

var p2=props.FirstOrDefault(p=>p.Name.Equals("MyType"));
p2.SetValue(m, Convert.ChangeType(1, p2.PropertyType), null);

这里为枚举ENUM赋值时,不会自动转换1成MyEnum.Test2,所以会异常。

加强版System.Convert.ChangeType如下:

public static class ConvertHelper
    {
        #region = ChangeType =
        public static object ChangeType(object obj, Type conversionType)
        {
            return ChangeType(obj, conversionType, Thread.CurrentThread.CurrentCulture);
        }
        public static object ChangeType(object obj, Type conversionType, IFormatProvider provider)
        {
            #region Nullable
            Type nullableType = Nullable.GetUnderlyingType(conversionType);
            if (nullableType != null)
            {
                if (obj == null)
                {
                    return null;
                }
                return Convert.ChangeType(obj, nullableType, provider);
            }
            #endregion
            if (typeof(System.Enum).IsAssignableFrom(conversionType))
            {
                return Enum.Parse(conversionType, obj.ToString());
            }
            return Convert.ChangeType(obj, conversionType, provider);
        }
        #endregion
    }

其中
为可空类型Nullable返回一个默认值。

尤其在使用PropertyInfo.SetValue方法时。

再次使文首的测试代码,但是使用强化版ConvertHelper,如下:

            Type t = typeof (model);
            PropertyInfo[] props = t.GetProperties();
            var m = new model();
            var p1 = props.FirstOrDefault(p => p.Name.Equals("StartDate"));
            p1.SetValue(m, ConvertHelper.ChangeType(DateTime.Now, p1.PropertyType), null);
            var p2 = props.FirstOrDefault(p => p.Name.Equals("MyType"));
            p2.SetValue(m, ConvertHelper.ChangeType(1, p2.PropertyType), null);

成功为m赋值。

赋空值的情况:

var p1=props.FirstOrDefault(p=>p.Name.Equals("StartDate"));
p1.SetValue(m, Convert.ChangeType(null, p1.PropertyType), null);

为什么要多此一举呢?因为我的需求中需要转换的对像全是字符串。还是上面的例子:

var p1 = props.FirstOrDefault(p => p.Name.Equals(“StartDate”)); p1.SetValue(m, DateTime.Now, null);
直接也能赋值成功,但是如果:

var p1 = props.FirstOrDefault(p => p.Name.Equals(“StartDate”)); p1.SetValue(m, “13/05/13”, null);
则会异常。

var p1 = props.FirstOrDefault(p => p.Name.Equals(“StartDate”)); p1.SetValue(m, ConvertHelper.ChangeType(“13/05/13”, p1.PropertyType), null);
而使用强化版ConvertHelper就能够成功执行。

方法:

 

public static T RequestQueryStringParam<T>(string key, T value)
{
    try
    {
        if (HttpContext.Current.Request.QueryString[key] == null)
            return (T)Convert.ChangeType(value, typeof(T), CultureInfo.InvariantCulture);
        return (T)Convert.ChangeType(HttpContext.Current.Request.QueryString[key], typeof(T), CultureInfo.InvariantCulture);
    }
    catch
    {
        return (T)Convert.ChangeType(value, typeof(T), CultureInfo.InvariantCulture);
    }
}
 
public static T RequestFromParam<T>(string key, T value)
{
    try
    {
        if (HttpContext.Current.Request.Form[key] == null)
            return (T)Convert.ChangeType(value, typeof(T), CultureInfo.InvariantCulture);
        return (T)Convert.ChangeType(HttpContext.Current.Request.Form[key], typeof(T), CultureInfo.InvariantCulture);
    }
    catch
    {
        return (T)Convert.ChangeType(value, typeof(T), CultureInfo.InvariantCulture);
    }
}
 
public static DateTime? RequestQueryStringParamDateTime(string key, DateTime? value)
{
    try
    {
        if (HttpContext.Current.Request.QueryString[key] == null)
            return null;
 
        if (!string.IsNullOrEmpty(HttpContext.Current.Request.QueryString[key]))
            return Convert.ToDateTime(HttpContext.Current.Request.QueryString[key]);
        return null;
    }
    catch
    {
        return null;
    }
}
 
public static DateTime? RequestFromParamDateTime(string key, DateTime? value)
{
    try
    {
        if (HttpContext.Current.Request.Form[key] == null)
            return null;
 
        if (!string.IsNullOrEmpty(HttpContext.Current.Request.Form[key]))
            return Convert.ToDateTime(HttpContext.Current.Request.Form[key]);
        return null;
    }
    catch
    {
        return null;
    }
}
 
#region = 加强版 ChangeType =
public static object ChangeType(object obj, Type conversionType)
{
    return ChangeType(obj, conversionType, Thread.CurrentThread.CurrentCulture);
}
public static object ChangeType(object obj, Type conversionType, IFormatProvider provider)
{
    #region Nullable
    Type nullableType = Nullable.GetUnderlyingType(conversionType);
    if (nullableType != null)
    {
        if (obj == null)
        {
            return null;
        }
        return Convert.ChangeType(obj, nullableType, provider);
    }
    #endregion
 
    #region Enum
    if (typeof(System.Enum).IsAssignableFrom(conversionType))
    {
        return Enum.Parse(conversionType, obj.ToString());
    }
    return Convert.ChangeType(obj, conversionType, provider);
    #endregion
}
#endregion

备注一下泛型的一些知识点:

什么是泛型?

C# 语言和公共语言运行时 (CLR) 在 2.0 版本中添加了泛型。泛型将类型参数的概念引入 .NET Framework,这样就可以设计具有相同特征的类和方法:在客户端代码声明并初始化这些类和方法之前,这些类和方法会延迟指定一个或多个类型(使用占位符~3:数字代表类型参数个数)。

泛型有什么用?

使用泛型类型可以最大限度地重用代码、保护类型安全性以及提高性能。泛型最常见的用途是创建集合类 List。

可以创建:泛型接口、泛型类、泛型方法、泛型事件和泛型委托。

泛型约束有哪些?

where T : struct 类型参数必须是值类型。

where T : class 类型参数必须是引用类型。 此约束还应用于任何类、接口、委托或数组类型。

where T : unmanaged 类型参数不能是引用类型,并且任何嵌套级别均不能包含任何引用类型成员。

where T : new() 类型参数必须具有公共无参数构造函数。

where T : <基类名> 类型参数必须是指定的基类或派生自指定的基类。

where T : <接口名称> 类型参数必须是指定的接口或实现指定的接口。

where T : U 为 T 提供的类型参数必须是为 U 提供的参数或派生自为 U 提供的参数。

某些约束是互斥的。 所有值类型必须具有可访问的无参数构造函数。 struct 约束包含 new() 约束,且 new() 约束不小蜜蜂论坛回帖机能与 struct 约束结合使用。 unmanaged 约束包含 struct 约束。 unmanaged 约束不能与 struct 或 new() 约束结合使用。

泛型类:

class GenericClass<T> { }

泛型接口:

public interface GenericInterface<T>{}

泛型方法:(可以作为传入参数,也可以作为返回值)

public T void Create<T>(T t)
{
    return default(T)
}

泛型委托:

public delegate void GenericDelegate<T>(T t);
  • 5
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值