<p style="white-space: normal;">BeanUtils.copyProperties深拷贝的使用。这里说的是 spring 的 BeanUtils.copyProperties。</p><p style="white-space: normal;"><strong>场景</strong></p><p style="white-space: normal;">开发中经常遇到,把父类的属性拷贝到子类中。通常有 2 种方法:</p><p style="white-space: normal;">1、一个一个 set;</p><p style="white-space: normal;">2、用 BeanUtils.copyProperties;很显然 BeanUtils 更加方便,也美观很多。那么任何情况都能使用 BeanUtils 么,当然不是。要先了解他。</p><p style="white-space: normal;"><strong>BeanUtils 是深拷贝,还是浅拷贝?</strong></p><p style="white-space: normal;">是浅拷贝。</p><p style="white-space: normal;">浅拷贝: 只是调用子对象的 set 方法,并没有将所有属性拷贝。(也就是说,引用的一个内存地址)</p><p style="white-space: normal;">深拷贝: 将子对象的属性也拷贝过去。</p><p style="white-space: normal;"><strong>什么情况适合用 BeanUtils</strong></p><p style="white-space: normal;">如果都是单一的属性,那么不涉及到深拷贝的问题,适合用 BeanUtils。有子对象就一定不能用 BeanUtils 么并不绝对,这个要区分考虑:</p><p style="white-space: normal;">1、子对象还要改动。</p><p style="white-space: normal;">2、子对象不怎么改动。</p><p style="white-space: normal;">虽然有子对象,但是子对象并不怎么改动,那么用 BeanUtils 也是没问题的。</p><p style="white-space: normal;"><strong>代码例子</strong></p><p style="white-space: normal;">下面用代码说明下。</p><p style="white-space: normal;">翠山有个儿子无忌,儿子继承了他的 face 和 height。</p><p style="white-space: normal;">但是 life 应该是自己的。</p><p style="white-space: normal;">后来翠山自刎而死,无忌也变成 dead 状态了。这就是浅拷贝,无忌用的 life 引用的翠山的 life 对象。</p><p style="white-space: normal;">Father 类:</p><pre class="brush:java;toolbar:false">@Data public class Father { private String face; // 长相 private String height; // 身高 private Life life; // 生命 }</pre><p style="white-space: normal;">Life 类:</p><pre class="brush:java;toolbar:false">@Data public class Life { private String status; }</pre><p style="white-space: normal;">Son 类和 main 方法:<br/></p><pre class="brush:java;toolbar:false">@Data public class Son extends Father { private Life life; public static void main(String[] args) { Father cuishan = new Father(); cuishan.setFace("handsome"); cuishan.setHeight("180"); Life cuishanLife = new Life(); cuishanLife.setStatus("alive"); cuishan.setLife(cuishanLife); Son wuji=new Son(); BeanUtils.copyProperties(cuishan,wuji); //Life wujiLife = wuji.getLife(); //wujiLife.setStatus("alive"); //wuji.setLife(wujiLife); //cuishanLife.setStatus("dead"); // 翠山后来自刎了 System.out.println(JSON.toJSONString(cuishan)); System.out.println(JSON.toJSONString(wuji)); } }</pre><p style="white-space: normal;">上面注释出的代码可以如下替换:<br/></p><p style="white-space: normal;">case1 和 case2 还是受浅拷贝的影响,case3 不受。</p><p style="white-space: normal;">case1: 翠山自刎,无忌也挂了</p><pre class="brush:java;toolbar:false">//Life wujiLife = wuji.getLife(); //wujiLife.setStatus("alive"); //wuji.setLife(wujiLife); //cuishanLife.setStatus("dead"); // 翠山后来自刎了</pre><p style="white-space: normal;">case2: 翠山自刎,无忌设置或者,翠山也活了</p><pre class="brush:java;toolbar:false">//cuishanLife.setStatus("dead"); // 翠山后来自刎了 //Life wujiLife = wuji.getLife(); //wujiLife.setStatus("alive"); //wuji.setLife(wujiLife);</pre><p style="white-space: normal;">case3: 翠山和无忌互不影响</p><pre class="brush:java;toolbar:false">cuishanLife.setStatus("dead"); // 翠山自刎了 该行放在上下均可 // 无忌用个新对象 不受翠山影响了 Life wujiLife = new Life(); wujiLife.setStatus("alive"); wuji.setLife(wujiLife);</pre><p style="white-space: normal;"><strong>BeanUtils.copyProperties深拷贝</strong></p><pre class="brush:java;toolbar:false">/** * 对象属性拷贝 <br> * 将源对象的属性拷贝到目标对象 * * @param source 源对象 * @param target 目标对象 */ public static void copyProperties(Object source, Object target) { try { BeanUtils.copyProperties(source, target); } catch (BeansException e) { LOGGER.error("BeanUtil property copy failed :BeansException", e); } catch (Exception e) { LOGGER.error("BeanUtil property copy failed:Exception", e); } }</pre><p style="white-space: normal;"><strong>BeanUtils.copyProperties深拷贝小结</strong></p><p style="white-space: normal;"><strong>1. 深拷贝和浅拷贝</strong></p><p style="white-space: normal;">深拷贝,相当于创建了一个新的对象,只是这个对象的所有内容,都和被拷贝的对象一模一样而已,即两者的修改是隔离的,相互之间没有影响</p><p style="white-space: normal;">浅拷贝,也是创建了一个对象,但是这个对象的某些内容(比如A)依然是被拷贝对象的,即通过这两个对象中任意一个修改A,两个对象的A都会受到影响,等同与新创建一个对象,然后使用=,将原对象的属性赋值给新对象的属性需要实现Cloneable接口。</p><p style="white-space: normal;"><strong>2. 对象拷贝的两种方法</strong></p><p style="white-space: normal;">通过反射方式实现对象拷贝,主要原理就是通过反射获取所有的属性,然后反射更改属性的内容。通过代理实现对象拷贝将原SourceA拷贝到目标DestB,创建一个代理 copyProxy。在代理中,依次调用 SourceA的get方法获取属性值,然后调用DestB的set方法进行赋值。</p>