java对象序列化去掉字段
假设您正在开发一个将对象自动保存到数据库中的框架。 您需要检测两次保存之间所做的更改,以便仅保存已修改的字段。 如何检测脏场。 最简单的方法是遍历原始数据和当前数据,并分别比较每个字段。 代码如下:
public static void getDirtyFields(Object obj, Object obj2, Class cls, Map<String, DiffFields> diff)
throws Exception {
Field[] flds = cls.getDeclaredFields();
for (int i = 0; i < flds.length; i++) {
flds[i].setAccessible(true);
Object fobj = flds[i].get(obj);
Object fobj2 = flds[i].get(obj2);
if (fobj.equals(fobj2)) continue;
if (checkPrimitive(flds[i].getType())) {
<!-- add to dirty fields -->
continue;
}
Map<String, DiffFields> fdiffs = new HashMap<String, DiffFields>();
getDirtyFields(fobj, fobj2, fobj.getClass(), fdiffs);
<!-- add to dirty fields -->
}
if (cls.getSuperclass() != null)
getDirtyFields(obj, obj2, cls.getSuperclass(), diff);
}
上面的代码不能处理很多条件,例如null值,字段是集合,映射或数组等。但是,这给出了可以做什么的想法。 如果对象很小并且其中不包含太多层次结构,则效果很好。 当在巨大的层次结构对象中的变化很小时,我们必须一直遍历到最后一个对象才能知道差异。 而且,使用equals可能不是检测脏字段的正确方法。 可能尚未实现等于,或者仅可以仅比较几个字段,所以没有进行真正的脏字段检测。 您必须遍历每个字段,而不论是否相等,直到您击中图元来检测脏字段为止。
在这里,我想谈谈检测脏场的另一种方法。 代替使用反射,我们可以使用序列化来检测脏字段。 我们可以轻松地替换上面代码中的“等于”来序列化对象,并且仅当字节不同时才继续操作。 但这不是最佳选择,因为我们将多次序列化同一对象。 我们需要如下逻辑:
- 序列化要比较的两个对象
- 比较两个字节流时,检测要比较的字段
- 如果字节值不同,则将该字段存储为不同
- 收集所有不同的字段并返回
因此,一次遍历两个字节流可以生成不同字段的列表。 我们如何实现这种逻辑? 我们可以遍历序列化流并能够识别其中的字段吗? 我们要编写如下代码:
public static void main(String[] args) throws Exception {
ComplexTestObject obj = new ComplexTestObject();
ComplexTestObject obj2 = new ComplexTestObject();
obj2._simple._string = "changed";
//serialize the first object and get the bytes
ByteArrayOutputStream ostr = new ByteArrayOutputStream();
CustomOutputS