淺拷貝與深拷貝——Java
阿新 • • 發佈:2020-12-12
List的深拷貝與淺拷貝:
一.淺拷貝
list本質是陣列,陣列時以地址的形式進行儲存
如上圖將list A淺拷貝給list B,由於進行的是淺拷貝,所以直接將A的內容複製給了B,java中相同內容的陣列指向同一地址,即進行淺拷貝後A與B指向同一地址。造成的後果就是,改變B的同時也會改變A,因為改變B就是改變B所指向地址的內容,由於A也指向同一地址,所以A與B一起改變。
淺拷貝方法:
1.遍歷迴圈複製
List<Person> destList=new ArrayList<Person>(srcList.size());
for (Person p : srcList){
destList.add(p);
}
2.使用List實現類的構造方法
List<Person> destList=new ArrayList<Person>(srcList);
3.使用list.addAll()方法
List<Person> destList=new ArrayList<Person>();
destList.addAll(srcList);
4.使用System.arraycopy()方法
Person[] srcPersons=srcList. toArray(new Person[0]);
Person[] destPersons=new Person[srcPersons.length];
System.arraycopy(srcPersons, 0, destPersons, 0, srcPersons.length);
二、深拷貝
如圖,深拷貝就是將A複製給B的同時,給B建立新的地址,再將地址A的內容傳遞到地址B。ListA與ListB內容一致,但是由於所指向的地址不同,所以改變相互不受影響。
深拷貝方法:
1.使用序列化方法
/**
* @Description: deepCopy,深度複製
* @param src: 複製物件集
* @return: java.util.List<T>
* @Author: hsy
* @Date: 2020/12/11
*/
public static <T> List<T> deepCopy(List<T> src) throws IOException, ClassNotFoundException {
ByteArrayOutputStream byteOut = new ByteArrayOutputStream();
ObjectOutputStream out = new ObjectOutputStream(byteOut);
out.writeObject(src);
ByteArrayInputStream byteIn = new ByteArrayInputStream(byteOut.toByteArray());
ObjectInputStream in = new ObjectInputStream(byteIn);
@SuppressWarnings("unchecked")
List<T> dest = (List<T>) in.readObject();
return dest;
}
2.clone方法
public class A implements Cloneable {
public String name[];
public A(){
name=new String[2];
}
public Object clone() {
A o = null;
try {
o = (A) super.clone();
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
return o;
}
}
for(int i=0;i<n;i+=){
copy.add((A)src.get(i).clone());
}
Java對物件和基本的資料型別的處理是不一樣的。在Java中用物件的作為入口引數的傳遞則預設為”引用傳遞”,也就是說僅僅傳遞了物件的一個”引用”,這個”引用”的概念同C語言中的指標引用是一樣的。當函式體內部對輸入變數改變時,實質上就是在對這個物件的直接操作。 除了在函式傳值的時候是”引用傳遞”,在任何用”=”向物件變數賦值的時候都是”引用傳遞”。
在淺複製的情況下,源資料被修改破壞之後,使用相同引用指向該資料的目標集合中的對應元素也就發生了相同的變化。因此,在需求要求必須深複製的情況下,要是使用上面提到的方法,請確保List中的T類物件是不易被外部修改和破壞的。
後續更新Map的深拷貝與淺拷貝。。。
參考:
https://blog.csdn.net/demonliuhui/article/details/54572908