Java 深度克隆clone 对象的拷贝(深层复制)

关于克隆(Object[clone])

克隆,主要作用是将当前对象进行复制拷贝,保留原有对象持有的属性、状态。是来自Object类中的clone方法,被关键字native修饰,是非java语言实现的接口。首先看一下Object中的clone方法,如下:
在这里插入图片描述

为什么使用克隆?克隆的场景是什么?

当程序运行中时,需要A对象的相同属性、状态来生成新的对象B,B对象与A对象具有相同的属性、状态。而A对象会继续执行后续的业务,同时后续的业务中也使用B对象。

克隆的分类

Java中的克隆,主要分为两类:

  1. 浅度克隆:只克隆对象的基本数据类型的属性,不会克隆引用类型的属性。当然,基本数据类型的包装类和String类型除外
  2. 深度克隆:克隆当前对象的所有持有属性(所有数据类型)。

克隆方式

1. 克隆对象的Class实现Cloneable接口

public class Student implements Cloneable {}

实现Cloneable接口后,方可使用Object中的clone方法。但是此方式只能克隆对象的基本数据类型(包括基本数据类型的包装类)和String类型的属性,属于浅度克隆。
如果要想通过此方式实现深度克隆,必须给克隆对象持有的引用类型的Class也实现Cloneable接口。代码如下:

Student类:

public class Student implements Cloneable {
	private int	stuId; // 学号
	private Teacher teacher; //持有Teacher对象
	// 克隆
	@Override
	public Object clone() throws CloneNotSupportedException {
		// 调用Object的clone方法,并强制转换为Student,得到新的Student对象stuClone
		Student stuClone = (Student)super.clone();
		// 从stuClone中获取teacher,这里的teacher并没有真正的实现克隆,只是持有引用。
		// 实际上,stuClone里的teacher引用指向的是原Student对象中teacher的内存堆地址。
		// 目前,原student对象与新的stuClone对象两者引用的Teacher是同一个Teacher对象。
		Teacher teacher = (Teacher)stuClone.getTeacher()
		// 现在,将从stuClone拿到的teacher对象再进行克隆,并将克隆到的teacherClone对象给stuClone
		Teacher teacherClone = teacher.clone();
		stuClone.setTeacher(teacherClone);
	}
	
	public Teacher getTeacher() {
		return teacher;
	}
	
	public void setTeacher(Teacher teacher) {
		this.teacher = teacher;
	}
}

Teacher类:


public class Teacher implements Cloneable {
	// 学科
	private String subject;
	// 克隆
	@Override
	public Object clone() throws CloneNotSupportedException {
		// 调用Object的clone方法
		Object obj = super.clone();
		return obj;
	}
	public String getSubject() {
		return subject;
	}
	
	public void setSubject(String subject) {
		this.subject = subject;
	}
}

小结: 实现Cloneable接口的方式,做浅度克隆还可以,但是要做深度克隆的话,需要手动地将对象的引用类型(基本数据类型的包装类和String类型除外)一个一个的克隆,维护起来很麻烦。本人不推荐使用。

2. 克隆对象的Class实现Serializable接口

public class Student implements Serializable {}

实现Serializable接口,利用序列化的方式实现克隆,这种方式是深度克隆。

其中原理:利用IO流方式,将对象写入流里,再从流里将对象读取出来(这一过程其实相当于复制拷贝)。实现Serializable的会将引用类型一并进行克隆。

为了方便,我这里使用了继承。父类:Person 子类:Student、Teacher。直接上代码:

Person类:

// 父类实现序列化接口Serializable,子类只需继承便可
public class Person implements Serializable {
	
	/** serialVersionUID */
	private static final long	serialVersionUID	= 1L;
	// 年龄
	private int					age;
	// 姓名
	private String				name;
	
	public Person() {}
	
	public Person(int age, String name) {
		this.age = age;
		this.name = name;
	}
	// 此方法是普通克隆方法
	public Person clone() {
		try {
			// 创建输出流,将对象放入流里
			ByteArrayOutputStream baos = new ByteArrayOutputStream();
			ObjectOutputStream oos = new ObjectOutputStream(baos);
			oos.writeObject(this);
			// 创建输入流,将对象读出来
			ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray());
			ObjectInputStream ois = new ObjectInputStream(bais);
			Person obj = (Person) ois.readObject();
			return obj;
		} catch (Exception e) {
			e.printStackTrace();
			return null;
		}
	}
	// 此方法是利用泛型的克隆方法,省去了最后的强制转换过程
	// 同时也限制了被克隆类的类型,必须继承Person
	public <T> T clone(Class<T> clazz) {
		try {
			// 如果目标类没有继承关系,可以将下面判断条件删除
			T newInstance = clazz.newInstance();
			if (!(newInstance instanceof Person)) {
				// 如果不是Person类型,抛异常
				throw new Exception("被clone的对象不属于基类Person类型");
			}
			// 创建输出流,将对象放入流里
			ByteArrayOutputStream baos = new ByteArrayOutputStream();
			ObjectOutputStream oos = new ObjectOutputStream(baos);
			oos.writeObject(this);
			// 创建输入流,将对象读出来
			ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray());
			ObjectInputStream ois = new ObjectInputStream(bais);
			Object readObject = ois.readObject();
			// 将对象转换为规定的泛型
			T obj = clazz.cast(readObject);
			return obj;
		} catch (Exception e) {
			e.printStackTrace();
			return null;
		}
	}
	
	public int getAge() {
		return age;
	}
	
	public void setAge(int age) {
		this.age = age;
	}
	
	public String getName() {
		return name;
	}
	
	public void setName(String name) {
		this.name = name;
	}
	
}

Student类:

// 继承父类Person
public class Student extends Person {
	
	/** serialVersionUID */
	private static final long	serialVersionUID	= 1L;
	// 学号
	private int					stuId;
	// 持有应用类型Teacher对象
	private Teacher				teacher;
	
	public Student() {}
	
	public Student(int age, String name, int stuId) {
		super(age, name);
		this.stuId = stuId;
	}
	
	public int getStuId() {
		return stuId;
	}
	
	public void setStuId(int stuId) {
		this.stuId = stuId;
	}
	
	public Teacher getTeacher() {
		return teacher;
	}
	
	public void setTeacher(Teacher teacher) {
		this.teacher = teacher;
	}
	
	@Override
	public String toString() {
		return "Student [age=" + this.getAge()+ ",name=" + this.getName()+ ",stuId=" + stuId + ", teacher=" + teacher + "]";
	}
	
}

Teacher类:

public class Teacher extends Person {
	
	/** serialVersionUID */
	private static final long	serialVersionUID	= 1L;
	// 学科
	private String				subject;
	
	public Teacher() {
	}
	
	public Teacher(String subject) {
		this.subject = subject;
	}
	
	public String getSubject() {
		return subject;
	}
	
	public void setSubject(String subject) {
		this.subject = subject;
	}
	
	@Override
	public String toString() {
		return "Teacher [subject=" + subject + "]";
	}
	
}

测试:

public static void main(String[] args) {
		// 创建学生对象
		Student stu1 = new Student(10, "zhangsan", 1001);
		// 给学生安排一个体育老师
		stu1.setTeacher(new Teacher("体育"));
		System.out.println("学生:" + stu1.toString());
		// 克隆
		Student clone = stu1.clone(Student.class);
		System.out.println("克隆" + clone.toString());
		System.out.println("------------------修改后--------------------");
		// 修改克隆对象的属性
		clone.setAge(22);
		clone.setName("lisi");
		clone.setStuId(1002);
		clone.getTeacher().setSubject("语文");
		System.out.println("克隆对象:" + clone.toString());
		System.out.println("学生对象:" + stu1.toString());
	}

打印输出:

学生:Student [age=10,name=zhangsan,stuId=1001, teacher=Teacher [subject=体育]]
克隆:Student [age=10,name=zhangsan,stuId=1001, teacher=Teacher [subject=体育]]
------------------修改后--------------------
克隆对象:Student [age=22,name=lisi,stuId=1002, teacher=Teacher [subject=语文]]
学生对象:Student [age=10,name=zhangsan,stuId=1001, teacher=Teacher [subject=体育]]

可以看出,克隆之后修改了克隆对象的基本类型和引用类型的属性,被克隆对象的属性并没有做任何修改。这就是深度克隆。

总结:两种方式实现了对象的克隆,针对深度克隆而言,实现Serializable这种方式要比实现Cloneable方式简单方便的多,而且易维护。
推荐使用Serializable!

原创文章,有不足之处还请各路大神多多指点,多多包涵。
如需转载,请标明出处,不胜感激。
                                                                                                        —— 码上仙

  • 6
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值