Object类的其中一个方法是拷贝方法clone()
,而拷贝可以分为深拷贝和浅拷贝,它们的区别主要在于对象中的引用拷贝后是否指向同一个对象。
浅拷贝
深拷贝
Java实现的深拷贝和浅拷贝
Java中Object.clone()
是浅拷贝,以下代码有深拷贝和浅拷贝两种方法:
public class User implements Cloneable {
public User() {}
public User(Integer id, String name) {
super();
this.id = id;
this.name = name;
}
private Integer id;
private String name;
private User creator;
public Integer getId() {
return id;
}
public void setId(Integer id) {
this.id = id;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public User getCreator() {
return creator;
}
public void setCreator(User creator) {
this.creator = creator;
}
@Override
public String toString() {
return "User " + super.toString() + " [id=" + id + ", name=" + name + ", creator=" + creator + "]";
}
// 浅拷贝
@Override
protected Object clone() throws CloneNotSupportedException {
return super.clone(); // 浅拷贝
}
/*
// 深拷贝
@Override
protected Object clone() throws CloneNotSupportedException {
User user = (User)super.clone();
if (user.getCreator() != null) {
user.setCreator((User)user.getCreator().clone());
}
return user;
}
*/
}
public class CloneDemo {
public static void main(String[] args) throws CloneNotSupportedException {
User user1 = new User();
user1.setId(1);
user1.setName("Nick Huang");
user1.setCreator(new User(2, "Viki"));
System.out.println(user1);
System.out.println(user1.clone());
}
}
用浅拷贝运行的日志,可以看到属性中的User对象地址是一样的:
User User@551ee3 [id=1, name=Nick Huang, creator=User User@545ec940 [id=2, name=Viki, creator=null]]
User User@1aeeb406 [id=1, name=Nick Huang, creator=User User@545ec940 [id=2, name=Viki, creator=null]]
用深拷贝运行的日志,可以看到属性中的User对象地址是不一样的:
User User@12dfbabd [id=1, name=Nick Huang, creator=User User@1e5cd7f9 [id=2, name=Viki, creator=null]]
User User@61672c01 [id=1, name=Nick Huang, creator=User User@3c7a279c [id=2, name=Viki, creator=null]]
上述是比较简单的方式实现深拷贝,另外有一些其他方式,比如序列化/反序列化
、转换为JSON/JSON转换回来
。
序列化的注意事项,有几个问题:
- 一个类实现Serializable,类里面的部分属性的类型没实现序列化,是否能序列化?答:不能,比如例子中,如果Role没有实现Serializable,对User序列化会报:
java.io.NotSerializableException: com.nicchagil.exercise.springbootexercise.WhichFieldWillSerializeTest$Role
- 子类实现Serializable、父类未实现Serializable,子类是否能序列化,子类中继承父类的属性是否能序列化?答:子类能序列化,但子类中继承父类的属性没有序列化