深拷贝和浅拷贝的基本概念的理解:
浅拷贝是指拷贝对象时仅仅拷贝对象本身(包括对象中的基本变量),而不拷贝对象包含的引用指向的对象,被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象。
深拷贝不仅拷贝对象本身,而且拷贝对象包含的引用指向的所有对象
Object.clone()实现
本身实现为浅拷贝
package com.suirui.common.util;
import com.suirui.common.constants.SysConstants;
import org.apache.commons.lang3.StringUtils;
public class TokenUtilTest {
public static void main(String[] args) {
Cat cat = new Cat("小花");
Person p1 = new Person("张三",cat);
try {
Person p2 =(Person) p1.clone();
p2.setName("李四");
Cat cat1 = p2.getCat();
cat1.setName("小白");
p2.setCat(cat1);
System.out.println(p1);
System.out.println(p2);
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
}
}
class Person implements Cloneable {
String name;
Cat cat;
public Cat getCat() {
return cat;
}
public void setCat(Cat cat) {
this.cat = cat;
}
public Person(String name,Cat cat) {
this.name = name;
this.cat = cat;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", cat=" + cat +
'}';
}
}
class Cat{
String name;
public Cat(String name) {
this.name = name;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "Cat{" +
"name='" + name + '\'' +
'}';
}
}
从结果可以看出来,clone()是浅copy。
实现深拷贝一
package com.suirui.common.util;
import com.suirui.common.constants.SysConstants;
import org.apache.commons.lang3.StringUtils;
public class TokenUtilTest {
public static void main(String[] args) {
Cat cat = new Cat("小花");
Person p1 = new Person("张三",cat);
try {
Person p2 =(Person) p1.clone();
p2.setName("李四");
Cat cat1 = p2.getCat();
cat1.setName("小白");
p2.setCat(cat1);
System.out.println(p1);
System.out.println(p2);
} catch (CloneNotSupportedException e) {
e.printStackTrace();
}
}
}
class Person implements Cloneable {
String name;
Cat cat;
public Cat getCat() {
return cat;
}
public void setCat(Cat cat) {
this.cat = cat;
}
public Person(String name,Cat cat) {
this.name = name;
this.cat = cat;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
protected Object clone() throws CloneNotSupportedException {
Person p = (Person) super.clone();
p.cat = (Cat) this.cat.clone();
return p;
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", cat=" + cat +
'}';
}
}
class Cat implements Cloneable{
String name;
public Cat(String name) {
this.name = name;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone();
}
@Override
public String toString() {
return "Cat{" +
"name='" + name + '\'' +
'}';
}
}
这种方式比较繁琐,要将所有引用类都进行克隆一次。
实现深拷贝二
在Java语言里深复制一个对象,常常可以先使对象实现Serializable接口,然后把对象(实际上只是对象的一个拷贝)写到一个流里,再从流里读出来,便可以重建对象。
注意:
写在流里的是对象的一个拷贝,而原对象仍然存在于JVM里面
。对象以及对象内部所有引用到的对象都是可序列化的
如果不想序列化,则需要使用transient来修饰
package com.suirui.common.util;
import com.suirui.common.constants.SysConstants;
import org.apache.commons.lang3.StringUtils;
import java.io.*;
public class TokenUtilTest {
public static void main(String[] args) {
Cat cat = new Cat("小花");
Person p1 = new Person("张三", cat);
try {
Person p2 = (Person) p1.clone();
p2.setName("李四");
Cat cat1 = p2.getCat();
cat1.setName("小白");
p2.setCat(cat1);
System.out.println(p1);
System.out.println(p2);
} catch (Exception e) {
e.printStackTrace();
}
}
}
class Person implements Serializable {
String name;
Cat cat;
public Cat getCat() {
return cat;
}
public void setCat(Cat cat) {
this.cat = cat;
}
public Person(String name, Cat cat) {
this.name = name;
this.cat = cat;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
protected Object clone() {
try {
//序列化对象
ByteArrayOutputStream bos = new ByteArrayOutputStream();
ObjectOutputStream oos = new ObjectOutputStream(bos);
oos.writeObject(this);
//反序列化对象
ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
ObjectInputStream ois = new ObjectInputStream(bis);
return ois.readObject();
} catch (Exception e) {
e.printStackTrace();
return null;
}
}
@Override
public String toString() {
return "Person{" +
"name='" + name + '\'' +
", cat=" + cat +
'}';
}
}
class Cat implements Serializable {
String name;
public Cat(String name) {
this.name = name;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "Cat{" +
"name='" + name + '\'' +
'}';
}
}