java学习之路——对象的浅复制和深复制

来源:互联网 发布:咫尺网络 编辑:程序博客网 时间:2024/06/10 06:11

 

对象的浅复制和深复制有两种方法:

第一种:通过重写clone()方法实现

1.浅复制与深复制概念
⑴浅复制(浅克隆)
    被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象。换言之,浅复制仅仅复制所考虑的对象,而不复制它所引用的对象。
⑵深复制(深克隆)
    被复制对象的所有变量都含有与原来的对象相同的值,除去那些引用其他对象的变量。那些引用其他对象的变量将指向被复制过的新对象,而不再是原有的那些被引用的对象。换言之,深复制把要复制的对象所引用的对象都复制了一遍。
2.Java的clone()方法
⑴clone方法将对象复制了一份并返回给调用者。一般而言,clone()方法满足:
①对任何的对象x,都有x.clone() !=x//克隆对象与原对象不是同一个对象
②对任何的对象x,都有x.clone().getClass()= =x.getClass()//克隆对象与原对象的类型一样
③如果对象x的equals()方法定义恰当,那么x.clone().equals(x)应该成立。
⑵Java中对象的克隆
①为了获取对象的一份拷贝,我们可以利用Object类的clone()方法。
②在派生类中覆盖基类的clone()方法,并声明为public。
③在派生类的clone()方法中,调用super.clone()。
④在派生类中实现Cloneable接口。 

例如:

package com.lcq.clone;/** *  * 类名:CloneTest1 * 功能:对象的浅复制和深复制,通过继承Cloneable接口和重写clone()方法实现 * 时间: *  * @version 1.0  * @author lcq * */public class CloneTest1 {public static void main(String[] args) throws Throwable {Teacher teacher = new Teacher();teacher.setAge(20);teacher.setName("teacher zhang");Student st = new Student();st.setAge(20);st.setName("zhangsan");st.setTeacher(teacher);//打印出Student对象的属性值System.out.println("st age:" +st.getAge());System.out.println("st name:" +st.getName());System.out.println(st.getTeacher().getAge());System.out.println(st.getTeacher().getName());Student st2 = (Student)st.clone();//改变student的name值st2.setName("lisi");//改变teacher对象的name属性值        teacher.setName("teacher name change");System.out.println("st name:" + st.getName());System.out.println("st2 name:" + st2.getName());//打印出Student类对象的teacher属性值,其值没有改变。System.out.println(st2.getTeacher().getName());}}class Student implements Cloneable{//定义属性private int age;private String name;//包含一个Teacher对象的引用private Teacher teacher;public Teacher getTeacher() {return teacher;}public void setTeacher(Teacher teacher) {this.teacher = teacher;}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;}//重写父类的clone()方法,同时要保证其成员对象中也重写了父类的clone()方法@Overridepublic Object clone() throws CloneNotSupportedException {Student stu = (Student) super.clone();stu.setTeacher((Teacher)stu.getTeacher().clone());return stu;}}class Teacher implements Cloneable{//定义属性private int age;private String name;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;}//重写父类的clone()方法@Overrideprotected Object clone() throws CloneNotSupportedException {Object obj = super.clone();return obj;}}


上图为浅复制。


上图为深复制。

第二种:实现深复制的方法是通过串行化和反串行化

利用对象串行化的特性:一个对象如果进行串行化则会将它所引用的对象都进行串行化。所以它的这个特性正好符合深度复制的要求。

例如以下示例:

package com.lcq.clone;/** *  * 类名:CloneTest2 * 功能:利用对象的序列化和反序列化实现对象的深复制 * 时间: *  * @version 1.0  * @author lcq * */import java.io.ByteArrayInputStream;import java.io.ByteArrayOutputStream;import java.io.ObjectInputStream;import java.io.ObjectOutputStream;import java.io.Serializable;public class CloneTest2 {public static void main(String[] args) throws Exception {Teacher2 teacher2 = new Teacher2();teacher2.setAge(40);teacher2.setName("teacher zhang");Student2 stu2 = new Student2();stu2.setAge(20);stu2.setName("student zhangsan");stu2.setTeacher2(teacher2);System.out.println("------------最初的学生对象信息-----------------");System.out.println(stu2.getAge());System.out.println(stu2.getName());System.out.println(stu2.getTeacher2().getAge());System.out.println(stu2.getTeacher2().getName());System.out.println("------------复制的学生对象信息-----------------");Student2 stu3 = (Student2)stu2.deepClone();System.out.println(stu3.getAge());System.out.println(stu3.getName());System.out.println(stu3.getTeacher2().getAge());System.out.println(stu3.getTeacher2().getName());System.out.println("------------修改复制对象后原学生对象信息-----------------");stu3.getTeacher2().setAge(30);stu3.getTeacher2().setName("teacher change");System.out.println(stu2.getTeacher2().getAge());System.out.println(stu2.getTeacher2().getName());}}class Teacher2 implements Serializable{/** *  */private static final long serialVersionUID = 502125658040435679L;private int age;private String name;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;}}class Student2 implements Serializable{/** *  */private static final long serialVersionUID = -3893457198772401952L;private int age;private String name;private Teacher2 teacher2;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;}public Teacher2 getTeacher2() {return teacher2;}public void setTeacher2(Teacher2 teacher2) {this.teacher2 = teacher2;}//定义深复制方法public Object deepClone() throws Exception{//将当前对象序列化ByteArrayOutputStream bao = new ByteArrayOutputStream();ObjectOutputStream oos = new ObjectOutputStream(bao);oos.writeObject(this);//将当前对象反序列化ByteArrayInputStream bai = new ByteArrayInputStream(bao.toByteArray());ObjectInputStream oi = new ObjectInputStream(bai);return oi.readObject();}}


原创粉丝点击