最近发现了一个现象:用BeanUtils.copyProperties(Object source, Object target)
时, 如果source和target中有List
对象,然后修改了source的List
对象中的元素,发现target的List
对象也会跟着变化。
调研了
BeanUtils.copyProperties(Object source, Object target)
源码,发现是通过method.invoke
方法来实现属性的copy的,所以对象的属性复制都是浅Copy。
于是乎就换了一个种方式来写:
List<A> source = new ArrayList<>();
source.add(new A());
List<A> copy = new ArrayList<>(source);
复制代码
发现居然还是不行!
觉得很奇怪,然后看了ArrayList
的构造方法,
public ArrayList(Collection<? extends E> c) {
elementData = c.toArray();
if ((size = elementData.length) != 0) {
// c.toArray might (incorrectly) not return Object[] (see 6260652)
if (elementData.getClass() != Object[].class)
elementData = Arrays.copyOf(elementData, size, Object[].class);
} else {
// replace with empty array.
this.elementData = EMPTY_ELEMENTDATA;
}
}
复制代码
elementData = c.toArray()
这段代码还是浅Copy。
思考了一会,难道只能通过遍历元素,然后添加元素的方式来解决吗?抱着疑问去Google了下,发现该函数Collections.copy
解决了问题,进入其源码发现是在遍历元素然后再添加。。。
总结
- copy完之后,一般都是直接返回给调用方了,一般不会再次操作数据,这样就不会引发浅Copy的问题。这也是copy的正确用法。