反射给对象赋值遇到的问题——类型转换
给一个对象属性赋值可以通过PropertyInfo.SetValue()方式进行赋值,但要注意值的类型要与属性保持一致。
创建对象实例的两种方法:
1.
1 | var obj = Assembly.Load( "AssemblyName" ).CreateInstance( "AssemblyName" + "ClassFullName" ); |
2.
1 | var obj = Activator.CreateInstance(ClassType); |
以后有时间再把这两种的区别详细讲一下。
创建好实例时,现在可以给当前实例的某个属性赋值,首先获取要赋值的属性。
1 | var property = obj.GetType().GetProperty( "PropertyName" ); //此时可以使用GetProperty获取属性数组,循环进行赋值,这里主要讲解类型问题。 |
赋值可通过方法,详见MSDN。
情况1,该属性类型是已知类型,例如:int
1 2 | int value=500; property.SetValue(obj,value, null ); |
这里需要注意value值的类型必须和属性类型一致,否则会抛出异常。
情况2,该属性类型是已知类型,原值是其他类型。例如:目标类型为int,值为string
1 2 | string value= "500" ; property.SetValue(obj, int .TryParse(value), null ); //类型转换。 |
前两种情况都很简单,有时业务会比较复杂,对目标类型不确定,需要程序运行时判断。
情况3,该属性类型是未知非泛型类型,不确定目标类型,如何进行类型转换。
1 2 | object value= "500" ; property.SetValue(obj,Convert.ChangeType(value,property.PropertyType), null ); //类型转换。 |
这样就可以解决大多数问题了。
不知道大家有没有注意,我在第三种情况强调了非泛型,难道泛型就不行了吗?
是的。如果只是用Convert.ChangeType()方法,类型转换仍然报错,先看下面的代码。即使目标类型和值的类型是一致,通过Convert.ChangeType()进行转换仍然报错。
解决这个问题,就要先把属性值类型转成基类型后,在进行Convert转换。看代码这样,在使用Convert.ChangeType()转换可空类型时,就不会报错了。
再增加一些基础的判断验证,代码就比较完善了。 1 2 3 4 5 6 7 8 9 10 11 12 13 14 | if (!property.PropertyType.IsGenericType) { //非泛型 property.SetValue(obj, string .IsNullOrEmpty(value) ? null : Convert.ChangeType(value, property.PropertyType), null ); } else { //泛型Nullable<> Type genericTypeDefinition = property.PropertyType.GetGenericTypeDefinition(); if (genericTypeDefinition == typeof (Nullable<>)) { property.SetValue(obj, string .IsNullOrEmpty(value) ? null : Convert.ChangeType(value, Nullable.GetUnderlyingType(property.PropertyType)), null ); } } |