【计算机基础】对象的深拷贝和浅拷贝

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,子类是否能序列化,子类中继承父类的属性是否能序列化?答:子类能序列化,但子类中继承父类的属性没有序列化
原文地址:https://www.cnblogs.com/nick-huang/p/6657574.html