我目前正在从事一个项目,在该项目中,要将字节数据流映射到输入类的属性。下面的反序列化代码可以完成其工作,但是我对DeserializeArray()-方法不满意。是否有更好和更清洁的方法来以反射的方式处理数组和非数组类型。
protected virtual void DoDeserialize(Type aType, object aInstance, PlcBinaryReader aReader)
{
foreach (PropertyInfo p in aType.GetProperties(BindingFlags.Public | BindingFlags.Instance))
{
if (p.GetValue(aInstance, null) != null)
{
if (p.PropertyType.IsArray)
DeserializeArray(p, aInstance, aReader); // this method breaks in my eyes the code
else
{
if (p.PropertyType.IsBasicType())
p.SetValue(aInstance, aReader.ReadData(p.PropertyType.Name, GetStrLength(p, false)));
else
DoDeserialize(p.PropertyType, p.GetValue(aInstance, null), aReader);
}
}
}
}
private void DeserializeArray(PropertyInfo aPropInfo, object aInstance, PlcBinaryReader aReader)
{
Array a = (Array)aPropInfo.GetValue(aInstance, null);
IList arr = a;
for (var i = 0; i < a.GetLength(0); i++)
{
Type t = aPropInfo.PropertyType.GetElementType();
if (t.IsBasicType())
arr[i] = aReader.ReadData(t.Name, GetStrLength(aPropInfo, true));
else
DoDeserialize(t, arr[i], aReader);
}
}
private byte GetStrLength(PropertyInfo aPropInfo, bool isArray)
{
byte result = 0;
Type t = isArray ? aPropInfo.PropertyType.GetElementType() : aPropInfo.PropertyType;
if (t.Name == "String")
{
var attr = aPropInfo.GetCustomAttributes(typeof(AdsStringLengthAttribute), true)
.OfType<AdsStringLengthAttribute>().SingleOrDefault();
result = attr != null ? attr.AdsStringLength : Constants.StdAdsStringLength;
}
return result;
}
tia