一、淺拷貝與深拷貝
在Java中,對象拷貝可以分為淺拷貝和深拷貝。淺拷貝即拷貝對象的引用地址,使得指向同一塊內存地址,而深拷貝則是在內存中重新申請一塊與源對象完全相同的內存空間,使得複製對象和原始對象互不影響。
public class Person implements Cloneable {
private String name;
private int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
public class Test {
public static void main(String[] args) throws CloneNotSupportedException {
// 淺拷貝
Person person1 = new Person("Tom", 18);
Person person2 = person1;
person1.setAge(20);
System.out.println(person1.getAge()); //20
System.out.println(person2.getAge()); //20
// 深拷貝
Person person3 = new Person("Jerry", 22);
Person person4 = (Person) person3.clone();
person3.setAge(25);
System.out.println(person3.getAge()); //25
System.out.println(person4.getAge()); //22
}
}
二、利用Cloneable接口實現淺拷貝
Java中的Cloneable接口提供了對象的淺拷貝機制,只需要在對象上實現Cloneable接口並重寫Object類中的clone方法即可。需要注意的是,與Serializable不同,Cloneable並不是只是用為標識用,而是真正地在底層實現了對象的淺拷貝。
public class Person implements Cloneable {
private String name;
private int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
}
public class Test {
public static void main(String[] args) throws CloneNotSupportedException {
Person person1 = new Person("Tom", 18);
Person person2 = (Person) person1.clone();
person1.setAge(20);
System.out.println(person1.getAge()); //20
System.out.println(person2.getAge()); //18
}
}
三、利用序列化實現深拷貝
與淺拷貝不同的是,深拷貝需要重新申請一塊完全相同的內存空間,在Java中,可以通過序列化來實現。
public class Person implements Serializable {
private String name;
private int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public Person deepClone() throws IOException, ClassNotFoundException {
ByteArrayOutputStream byteOut = new ByteArrayOutputStream();
ObjectOutputStream objOut = new ObjectOutputStream(byteOut);
objOut.writeObject(this);
ByteArrayInputStream byteIn = new ByteArrayInputStream(byteOut.toByteArray());
ObjectInputStream objIn = new ObjectInputStream(byteIn);
return (Person) objIn.readObject();
}
}
public class Test {
public static void main(String[] args) throws IOException, ClassNotFoundException {
Person person1 = new Person("Tom", 18);
Person person2 = person1.deepClone();
person1.setAge(20);
System.out.println(person1.getAge()); //20
System.out.println(person2.getAge()); //18
}
}
四、第三方庫實現深拷貝
除了利用序列化之外,Java中也有一些第三方庫可以方便地實現深拷貝。
public class Person {
private String name;
private int age;
public Person(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
public static T deepClone(T src) {
return SerializationUtils.clone(src);
}
}
public class Test {
public static void main(String[] args) {
Person person1 = new Person("Tom", 18);
Person person2 = Person.deepClone(person1);
person1.setAge(20);
System.out.println(person1.getAge()); //20
System.out.println(person2.getAge()); //18
}
}
五、結論
對象拷貝可以分為淺拷貝和深拷貝兩種方式,淺拷貝不會創建新的對象實例,而是指向同一個對象實例;深拷貝會創建一個新的對象實例,並且將原對象中所有的屬性都複製到新對象中。實現對象拷貝有多種方式,可以利用Cloneable接口和重寫clone方法來實現淺拷貝,可以利用序列化來實現深拷貝,也可以使用第三方庫來方便地實現深拷貝。
原創文章,作者:PKFC,如若轉載,請註明出處:https://www.506064.com/zh-hant/n/131660.html
微信掃一掃
支付寶掃一掃