【Java对象拷贝机制】使用CGlib实现Bean拷贝(BeanCopier)

简介: 【Java对象拷贝机制】使用CGlib实现Bean拷贝(BeanCopier)

对象拷贝现状


业务系统中经常需要两个对象进行属性的拷贝,不能否认逐个的对象拷贝是最快速最安全的做法,但是当数据对象的属性字段数量超过程序员的容忍的程度,代码因此变得臃肿不堪,使用一些方便的对象拷贝工具类将是很好的选择。



模型数据转换


项目中或多或少会对某些实体进行转换(DTO、VO、DO或者PO等),往往具有相同的属性名称,数量少的情况下我们可以直接采取set、get方法进行赋值,可是如果这样的转换在很多地方都会用到,还是靠set来进行操作势必会大大的影响开发效率。


  • 关于实体转换,我们把一个实体对应一张表(这可以当成DO)。
  • 业务中与第三方进行数据交互,我们需要把实体的数据传给他们,但不一定是一个DO中的所有属性可能减少或者多个DO中的属性组成,这里我们引入DTO(这个实体中我们可以去除一些隐私信息,比如:银行卡号,身份证,密码)。
  • 一个性别我们用1、2表示男女,页面中不能直接显示1或者2,需要显示男、女或者靓仔(男)、靓妹(女),这时候代表这样的一个实体我们可以看作VO。




目前流行的较为公用认可的工具类:


Apache的两个版本:(反射机制)


  • org.apache.commons.beanutils.PropertyUtils.copyProperties(Object dest, Object orig)


原因:dateTimeConveter的conveter没有对null值的处理

// targetObject特殊属性的限制:(Date,BigDecimal等)
public class BeanObject { //此处省略getter,setter方法
    private String name;
    private java.util.Date date;
}
 public class BeanObjectTest {  
    public static void main(String args[]) throws Throwable  {  
    BeanObject from = new BeanObject(); 
    BeanObject to = new BeanObject();  
    //from.setDate(new java.util.Date());
    from.setName("TTTT");
    org.apache.commons.beanutils.BeanUtils.copyProperties(to, from);//如果from.setDate去掉,此处出现conveter异常
    System.out.println(ToStringBuilder.reflectionToString(from));
    System.out.println(ToStringBuilder.reflectionToString(to));
    }  
}
复制代码



  • org.apache.commons.beanutils.BeanUtils.copyProperties(Object dest, Object orig)
  • 相同属性名,且类型不匹配时候的处理
  • 原因:这两个工具类不支持同名异类型的匹配 !!!【包装类Long和原始数据类型long是可以的】
public class SourceClass {  //此处省略getter,setter方法
    private Long num;  
    private String name;
}
public class TargetClass {  //此处省略getter,setter方法
    private Long num;
    private String name;
}
public class PropertyUtilsTest {
    public static void main(String args[]) throws IllegalAccessException, InvocationTargetException, NoSuchMethodException  {
        SourceClass from = new SourceClass();
        from.setNum(1);
        from.setName("name");
        TargetClass to = new TargetClass();
        //抛出参数不匹配异常
        org.apache.commons.beanutils.PropertyUtils.copyProperties(to, from);
        org.springframework.beans.BeanUtils.copyProperties(from, to);
        //抛出参数不匹配异常
        System.out.println(ToStringBuilder.reflectionToString(from));
        System.out.println(ToStringBuilder.reflectionToString(to));
    }
}
复制代码



Spring版本:(反射机制)


  • org.springframework.beans.BeanUtils.copyProperties(Object source, Object target, Class editable, String[] ignoreProperties)


cglib版本:(使用动态代理,效率高)


cglib是一款比较底层的操作java字节码的框架


  • net.sf.cglib.beans.BeanCopier.copy(Object paramObject1, Object paramObject2, Converter paramConverter)


工具操作

image.png



原理简介


反射类型:(apache)


都使用静态类调用,最终转化虚拟机中两个单例的工具对象。


public BeanUtilsBean(){
  this(new ConvertUtilsBean(), new PropertyUtilsBean());
}
复制代码


  • ConvertUtilsBean可以通过ConvertUtils全局自定义注册。
  • ConvertUtils.register(new DateConvert(), java.util.Date.class);
  • PropertyUtilsBean的copyProperties方法实现了拷贝的算法。
  1. 动态bean:orig instanceof DynaBean:Object value = ((DynaBean)orig).get(name); 然后把value复制到动态bean类。
  2. Map类型:orig instanceof Map:key值逐个拷贝
  3. 其他普通类:从beanInfo【每一个对象都有一个缓存的bean信息,包含属性字段等】取出name,然后把sourceClass和targetClass逐个拷贝。



Cglib类型:BeanCopier

copier = BeanCopier.create(source.getClass(), target.getClass(), false);
copier.copy(source, target, null);
复制代码



Get和set方法不匹配的处理

public class BeanCopierTest {
    /**
     * 从该用例看出BeanCopier.create的target.class 的每一个get方法必须有队形的set方法
     * @param args
     */
    public static void main(String args[]) {  
        BeanCopier copier = BeanCopier.create(UnSatifisedBeanCopierObject.class, SourceClass.class,false);
        copier = BeanCopier.create(SourceClass.class, UnSatifisedBeanCopierObject.class, false); //此处抛出异常创建 
    }  
}
class UnSatifisedBeanCopierObject {   
    private String name;
    private Long num;
    public String getName() {undefined
       return name;
    }
    public void setName(String name) {undefined
       this.name = name;
    }
    public Long getNum() {undefined
       return num;
    }
//  public void setNum(Long num) {undefined
//     this.num = num;
//  }
}
复制代码


Create对象过程:产生sourceClass-> TargetClass 的拷贝代理类,放入jvm中,所以创建的代理类的时候比较耗时。最好保证这个对象的单例模式,可以参照最后一部分的优化方案。



创建过程 -> 源代码见jdk:


net.sf.cglib.beans.BeanCopier.Generator.generateClass(ClassVisitor)


  1. 获取sourceClass的所有public get 方法-》PropertyDescriptor[] getters
  2. 获取TargetClass 的所有 public set 方法-》PropertyDescriptor[] setters
  3. 遍历setters的每一个属性,执行4和5
  4. 按setters的name生成sourceClass的所有setter方法-》PropertyDescriptor getter【不符合javabean规范的类将会可能出现空指针异常】
  5. PropertyDescriptor[] setters-》PropertyDescriptor setter
  6. 将setter和getter名字和类型 配对,生成代理类的拷贝方法。



原理总结


Copy属性过程:调用生成的代理类,代理类的代码和手工操作的代码很类似,效率非常高。


上述这几种方式速度最快的是BeanCopier,默认只复制名称和类型相同的字段,还会对date为空的情况不进行复制。


我认为这样做最好,比如对象A的值复制到B中,我们把相同的进行复制,把不同的,也就是需要我们个性化的一些字段,单独出来用get来赋值,这样程序就会很明确,重点也就聚焦在了不同的地方。




相关文章
|
15天前
|
安全 Java 编译器
Java对象一定分配在堆上吗?
本文探讨了Java对象的内存分配问题,重点介绍了JVM的逃逸分析技术及其优化策略。逃逸分析能判断对象是否会在作用域外被访问,从而决定对象是否需要分配到堆上。文章详细讲解了栈上分配、标量替换和同步消除三种优化策略,并通过示例代码说明了这些技术的应用场景。
Java对象一定分配在堆上吗?
|
19天前
|
Java API
Java 对象释放与 finalize 方法
关于 Java 对象释放的疑惑解答,以及 finalize 方法的相关知识。
39 17
|
9天前
|
消息中间件 缓存 Java
java nio,netty,kafka 中经常提到“零拷贝”到底是什么?
零拷贝技术 Zero-Copy 是指计算机执行操作时,可以直接从源(如文件或网络套接字)将数据传输到目标缓冲区, 而不需要 CPU 先将数据从某处内存复制到另一个特定区域,从而减少上下文切换以及 CPU 的拷贝时间。
java nio,netty,kafka 中经常提到“零拷贝”到底是什么?
|
18天前
|
存储 安全 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第22天】在Java的世界里,对象序列化和反序列化是数据持久化和网络传输的关键技术。本文将带你了解如何在Java中实现对象的序列化与反序列化,并探讨其背后的原理。通过实际代码示例,我们将一步步展示如何将复杂数据结构转换为字节流,以及如何将这些字节流还原为Java对象。文章还将讨论在使用序列化时应注意的安全性问题,以确保你的应用程序既高效又安全。
|
27天前
|
存储 Java 数据管理
Java零基础-Java对象详解
【10月更文挑战第7天】Java零基础教学篇,手把手实践教学!
23 6
|
1月前
|
Oracle Java 关系型数据库
重新定义 Java 对象相等性
本文探讨了Java中的对象相等性问题,包括自反性、对称性、传递性和一致性等原则,并通过LaptopCharger类的例子展示了引用相等与内容相等的区别。文章还介绍了如何通过重写`equals`方法和使用`Comparator`接口来实现更复杂的相等度量,以满足特定的业务需求。
18 3
|
1月前
|
存储 Java
Java编程中的对象序列化与反序列化
【10月更文挑战第9天】在Java的世界里,对象序列化是连接数据持久化与网络通信的桥梁。本文将深入探讨Java对象序列化的机制、实践方法及反序列化过程,通过代码示例揭示其背后的原理。从基础概念到高级应用,我们将一步步揭开序列化技术的神秘面纱,让读者能够掌握这一强大工具,以应对数据存储和传输的挑战。
|
1月前
|
XML Java Maven
在 Cucumber 测试中自动将 Cucumber 数据表映射到 Java 对象
在 Cucumber 测试中自动将 Cucumber 数据表映射到 Java 对象
50 7
|
1月前
|
存储 Java 数据管理
Java零基础-Java对象详解
【10月更文挑战第3天】Java零基础教学篇,手把手实践教学!
14 1
|
1月前
|
Java 数据安全/隐私保护
java类和对象
java类和对象
23 5