引言
在业务系统的开发需求中经常需要记录数据的变更日志,并显示给管理人员查看,可能的需求包括:
1、记录每一次的数据修改、数据中的每一项变更:变更时间、变更人、变更项等。
2、在页面中显示数据中的每一项变更:变更项的名称、变更前后的值。
这里边隐含了一些不太明显的需求,比如:
1、数据中某些项不需要记录变更,比如:修改时间。
2、数据中保存的是内部编号,需要显示可读的描述信息给管理人员,比如:城市ID。
3、数据项一般都是英文名称,需要显示中文信息给管理人员,比如:Gender。
另外有时候由于设计方面的原因,某个业务模型可能包含多个或者多级子类型,这时候进行比较就会涉及到嵌套的问题,比较复杂。
如果没有一个较好的处理模式或者处理工具,这个任务将带来不小的工作量,有时候甚至需要在前后台进行大量的关于此项的工作。当然如果不存在这些隐含的需求,或者这些需求可以被忽略,那就没有这么多事了,但是开发人员一般都是被指挥的。
ModelDataCompare
ModelDataCompare 提供了一组方法用于比较同一个类型的两个实例之间的数据差异。 常用于计算某种业务数据的变更,比如“用户信息”修改后计算修改了哪些用户的属性、属性修改前后的新旧值。
重要的是ModelDataCompare是完全免费并开源的,中文注释,提供详细的测试用例,在这里可以下载:
https://github.com/bosima/ModelDataCompare
下边就来先熟悉下这个工具:
ModelDataCompare中数据变更的结果通过ChangeItem定义:
public class ChangeItem
{
private IList<ChangeItem> _sub;
/// <summary>
/// 变更项所属类型的名称
/// </summary>
public string OfTypeName { get; set; }
/// <summary>
/// 变更项类型的名称
/// </summary>
public string TypeName { get; set; }
/// <summary>
/// 变更项名称
/// </summary>
public string Name { get; set; }
/// <summary>
/// 变更项业务人员可读名称
/// </summary>
public string ReadableName { get; set; }
/// <summary>
/// 修改前值
/// </summary>
public string OldValue { get; set; }
/// <summary>
/// 修改后值
/// </summary>
public string NewValue { get; set; }
/// <summary>
/// 如果是一个复杂类型,则可能具有下级变更项
/// </summary>
public IList<ChangeItem> Sub
{
get
{
if (_sub == null)
{
_sub = new List<ChangeItem>();
}
return _sub;
}
set
{
_sub = value;
}
}
}
ModelDataCompare 不仅仅是简单的数据比对,还提供了若干增强支持:
1、嵌套类型比较:
如下定义的模型类,其中的User属性也是一个复杂数据类型,ModelDataCompare 可以对User中的每一个属性进行比较。
/// <summary>
/// 嵌套类型
/// </summary>
public class NestedModel
{
public SimpleModel User
{
get;
set;
}
public string DeptName
{
get;
set;
}
public int? Year
{
get;
set;
}
public string Address
{
get;
set;
}
}
2、集合类型比较:
ModelDataCompare 可以比较集合中对应索引位置的同一个类型的实例,还支持集合元素数量不相等的比较。
SimpleModel[] oldArray = new SimpleModel[]{
new SimpleModel()
{
Name = "张三",
Age = 24,
Gender = 1,
City = 1,
Province = 1
},
new SimpleModel()
{
Name = "李四",
Age = 25,
Gender = 1,
City = 2,
}
};
SimpleModel[] newArray = new SimpleModel[]{
new SimpleModel()
{
Name = "张三",
Age = 23,
Gender = 1,
City = 1,
Province = 1
},
new SimpleModel()
{
Name = "李四",
Age = 24,
Gender = 1,
City = 3,
}
};
var changeItem = Comparer.Compare(oldArray, newArray, null, null);
3、数据属性友好名称
程序中的属性一般使用英文名称,对于业务系统的使用用户来说可读性较差,ModelDataCompare 支持为属性提供一个友好的可读名称。
var oldModel = new SimpleModel()
{
Name = "张三",
Age = 24,
Gender = 1,
City = 1,
Province = 1,
ID = Guid.NewGuid()
};
var newModel = new SimpleModel()
{
Name = "李四",
Age = 25,
Gender = 0,
City = 2,
ID = Guid.NewGuid()
};
var propertyReadableNameDictionary = new Dictionary<string, string>();
propertyReadableNameDictionary.Add("Name", "姓名");
propertyReadableNameDictionary.Add("Age", "年龄");
propertyReadableNameDictionary.Add("Gender", "性别");
propertyReadableNameDictionary.Add("City", "城市");
propertyReadableNameDictionary.Add("Province", "省");
var changeItem = Comparer.Compare(oldModel, newModel, new string[] { "ID" }, propertyReadableNameDictionary);
changeItem.ReadableName = "简单类型";
Assert.AreEqual("姓名", changeItem.Sub.Where(i => i.Name == "Name").FirstOrDefault().ReadableName);
4、数据属性值友好描述
属性对应的值可能只是一个系统内部使用的编号,对于业务系统的使用用户来说看不懂,ModelDataCompare 支持通过特性的方式转换相应的内部值为一个用户可读的值。
目前特性支持两种方式:枚举、方法。
public class SimpleModel
{
public Guid ID { get; set; }
public string Name { get; set; }
public int Age { get; set; }
[ValueDescription(EnumType = typeof(EnumGender))]
public int? Gender { get; set; }
[ValueDescription(QueryMethod = "FireflySoft.ModelDataCompare.UnitTest,FireflySoft.ModelDataCompare.UnitTest.CityBLL,GetModel,CityName")]
public int? City { get; set; }
public int? Province { get; set; }
}
public enum EnumGender
{
[System.ComponentModel.Description("先生")]
Male = 1,
[System.ComponentModel.Description("女士")]
Female = 0
}
public class CityBLL
{
public CityModel GetModel(int cityID)
{
var model = new CityModel();
model.CityID = 1;
model.CityName = "CityName";
return model;
}
}
public class CityModel
{
public int CityID { get; set; }
public string CityName { get; set; }
}
5、排除无关项
排序不关心或者不重要的数据属性,比如:数据修改时间。
var changeItem = Comparer.Compare(oldModel, newModel, new string[] { "ID" }, propertyReadableNameDictionary);
以上5点基本解决了各种独特的需求。
ModelDataCompare 内部采用了反射的方式用于获取类型的属性、获取属性的特性,以及获取相关方法和属性的值,这对性能会造成一定的影响。为了解决这个问题,ModelDataCompare 内部对相关反射获取到的数据进行了缓存,可以提升一些性能。
有需要的朋友赶紧去GitHub看看吧。
关键字: ModelDataCompare 变更日志
发表评论
相关文章
国内AI资源汇总,AI聊天、AI绘画、AI写作、AI视频、AI设计、AI编程、AI音乐等,国内顺畅访问,无需科学上网。
扫码或点击进入:萤火AI大全
文章分类
最新评论