浅谈Java中的深拷贝和浅拷贝

引出问题

假如说你想复制一个简单变量。很简单:

		int apples = 5;
		int pears = apples;

不仅仅是int类型,其它七种原始数据类型(boolean,char,byte,short,float,double.long)同样适用于该类情况。

但是如果你复制的是一个对象,情况就有些复杂了。

假设说我是一个beginner,我会这样写:

//创建类
class Student {
	private int number;
 
	public int getNumber() {
		return number;
	}
 
	public void setNumber(int number) {
		this.number = number;
	}
	
}
public class Test {
	
	public static void main(String args[]) {
		
		Student stu1 = new Student();
		stu1.setNumber(12345);
		Student stu2 = stu1;   //stu1与stu2引用同一个对象
		
		System.out.println("学生1:" + stu1.getNumber());
		System.out.println("学生2:" + stu2.getNumber());
	}
}

打印结果:

学生1:12345
学生2:12345

这里我们自定义了一个学生类,该类只有一个number字段。

我们新建了一个学生实例,然后将该值赋值给stu2实例。(Student stu2 = stu1;)

再看看打印结果,作为一个新手,拍了拍胸腹,对象复制不过如此,

难道真的是这样吗?

我们试着改变stu2实例的number字段,再打印结果看看:

stu2.setNumber(54321);

System.out.println("学生1:" + stu1.getNumber());
System.out.println("学生2:" + stu2.getNumber());
学生1:54321
学生2:54321

这就怪了,为什么改变学生2的学号,学生1的学号也发生了变化呢?

原因出在(stu2 = stu1) 这一句。该语句的作用是将stu1的引用赋值给stu2,

这样,stu1和stu2指向内存堆中同一个对象。如图:
在这里插入图片描述
那么,怎样才能达到复制一个对象呢?

浅拷贝

是否记得万类之王Object。它有11个方法,有两个protected的方法,其中一个为clone方法。

该方法的签名是:

protected native Object clone() throws CloneNotSupportedException;

因为每个类直接或间接的父类都是Object,因此它们都含有clone()方法,但是因为该方法是protected,所以都不能在类外进行访问。

要想对一个对象进行复制,就需要对clone方法覆盖

一般步骤是(浅复制):

  1. 被复制的类需要实现Clonenable接口(不实现的话在调用clone方法会抛出CloneNotSupportedException异常) 该接口为标记接口(不含任何方法)

  2. 覆盖clone()方法,访问修饰符设为public。方法中调用super.clone()方法得到需要的复制对象,(native为本地方法)

下面对上面那个方法进行改造:

//被复制的类实现Cloneable接口
class Student implements Cloneable{
	private int number;
 
	public int getNumber() {
		return number;
	}
 
	public void setNumber(int number) {
		this.number = number;
	}
	
	@Override
	//重写clone()方法,
	public Object clone() {
		Student stu = null;
		try{
			stu = (Student)super.clone();   //注意,super.clone()前放置Student类
		}catch(CloneNotSupportedException e) {
			e.printStackTrace();
		}
		return stu;
	}
}
public class Test {
	
	public static void main(String args[]) {
		
		Student stu1 = new Student();
		stu1.setNumber(12345);
		//注意,这里赋值的方法也变了,同时前面有强制类型转换,因为stu1.clone()返回值是Object
		Student stu2 = (Student)stu1.clone();  
		
		System.out.println("学生1:" + stu1.getNumber());
		System.out.println("学生2:" + stu2.getNumber());
		
		stu2.setNumber(54321);
	
		System.out.println("学生1:" + stu1.getNumber());
		System.out.println("学生2:" + stu2.getNumber());
	}
}

打印结果:

学生1:12345
学生2:12345
学生1:12345
学生2:54321

如果你还不相信这两个对象不是同一个对象,那么你可以看看这一句:

System.out.println(stu1 == stu2); // false

深度拷贝(实现clone方法)

上面的复制被称为浅复制(Shallow Copy),还有一种稍微复杂的深度复制(deep copy),即要复制的对象里面的成员变量含有对象:

我们在学生类里再加一个Address类。

class Address  {
	private String add;
 
	public String getAdd() {
		return add;
	}
 
	public void setAdd(String add) {
		this.add = add;
	}
	
}
 
class Student implements Cloneable{
	private int number;
 
	private Address addr;
	
	public Address getAddr() {
		return addr;
	}
 
	public void setAddr(Address addr) {
		this.addr = addr;
	}
 
	public int getNumber() {
		return number;
	}
 
	public void setNumber(int number) {
		this.number = number;
	}
	
	@Override
	public Object clone() {
		Student stu = null;
		try{
			stu = (Student)super.clone();
		}catch(CloneNotSupportedException e) {
			e.printStackTrace();
		}
		return stu;
	}
}
public class Test {
	
	public static void main(String args[]) {
		
		Address addr = new Address();
		addr.setAdd("杭州市");
		Student stu1 = new Student();
		stu1.setNumber(123);
		stu1.setAddr(addr);
		
		Student stu2 = (Student)stu1.clone();
		
		System.out.println("学生1:" + stu1.getNumber() + ",地址:" + stu1.getAddr().getAdd());
		System.out.println("学生2:" + stu2.getNumber() + ",地址:" + stu2.getAddr().getAdd());
	}
}

打印结果:

学生1:123,地址:杭州市
学生2:123,地址:杭州市

乍一看没什么问题,真的是这样吗?
我们在main方法中试着改变addr实例的地址。

addr.setAdd("西湖区");

System.out.println("学生1:" + stu1.getNumber() + ",地址:" + stu1.getAddr().getAdd());
System.out.println("学生2:" + stu2.getNumber() + ",地址:" + stu2.getAddr().getAdd());

打印结果:

学生1:123,地址:杭州市
学生2:123,地址:杭州市
学生1:123,地址:西湖区
学生2:123,地址:西湖区

这就奇怪了,怎么两个学生的地址都改变了?
原因是浅复制只是复制了addr变量的引用,并没有真正的开辟另一块空间,将值复制后再将引用返回给新对象

所以,为了达到真正的复制对象,而不是纯粹引用复制。我们需要将Address类可复制化,并且修改clone方法,完整代码如下:

package abc;
 
class Address implements Cloneable {
	private String add;
 
	public String getAdd() {
		return add;
	}
 
	public void setAdd(String add) {
		this.add = add;
	}
	
	@Override
	public Object clone() {
		Address addr = null;
		try{
			addr = (Address)super.clone();
		}catch(CloneNotSupportedException e) {
			e.printStackTrace();
		}
		return addr;
	}
}
 
class Student implements Cloneable{
	private int number;
 
	private Address addr;
	
	public Address getAddr() {
		return addr;
	}
 
	public void setAddr(Address addr) {
		this.addr = addr;
	}
 
	public int getNumber() {
		return number;
	}
 
	public void setNumber(int number) {
		this.number = number;
	}
	
	@Override
	public Object clone() {
		Student stu = null;
		try{
			stu = (Student)super.clone();	//浅复制
		}catch(CloneNotSupportedException e) {
			e.printStackTrace();
		}
		stu.addr = (Address)addr.clone();	//深度复制
		return stu;
	}
}
public class Test {
	
	public static void main(String args[]) {
		
		Address addr = new Address();
		addr.setAdd("杭州市");
		Student stu1 = new Student();
		stu1.setNumber(123);
		stu1.setAddr(addr);
		
		Student stu2 = (Student)stu1.clone();
		
		System.out.println("学生1:" + stu1.getNumber() + ",地址:" + stu1.getAddr().getAdd());
		System.out.println("学生2:" + stu2.getNumber() + ",地址:" + stu2.getAddr().getAdd());
		
		addr.setAdd("西湖区");
		
		System.out.println("学生1:" + stu1.getNumber() + ",地址:" + stu1.getAddr().getAdd());
		System.out.println("学生2:" + stu2.getNumber() + ",地址:" + stu2.getAddr().getAdd());
	}
}

打印结果:

学生1:123,地址:杭州市
学生2:123,地址:杭州市
学生1:123,地址:西湖区
学生2:123,地址:杭州市

这样结果就符合我们的想法了。

深度拷贝(实现对象序列化)

除了这种通过实现clone方法实现深拷贝,还有一种方法:通过对象序列化实现深拷贝

import java.io.*;

class Address implements Serializable{
    private String add;

    public String getAdd() {
        return add;
    }

    public void setAdd(String add) {
        this.add = add;
    }
}

class Student implements Serializable{
    private int number;

    private Address addr;

    public Address getAddr() {
        return addr;
    }

    public void setAddr(Address addr) {
        this.addr = addr;
    }

    public int getNumber() {
        return number;
    }

    public void setNumber(int number) {
        this.number = number;
    }

    //对象序列化实现深拷贝
    public Object deepClone() throws IOException {
        //创建流对象
        ByteArrayOutputStream bos=null;
        ObjectOutputStream oos=null;
        ByteArrayInputStream bis=null;
        ObjectInputStream ois=null;

        try{
            //序列化
            bos=new ByteArrayOutputStream();
            //字节数组输出流->对象输出流
            oos=new ObjectOutputStream(bos);
            //将当前对象以对象流的方式输出,如果对象中包含对象的引用,那么也会序列化
            oos.writeObject(this);

            //反序列化
            //对象输出流将对象输出后,再序列化读取,相当于进行对象的克隆
            bis=new ByteArrayInputStream(bos.toByteArray());
            ois=new ObjectInputStream(bis);
            //获取克隆对象
            Student student =(Student) ois.readObject();
            return student;
        } catch (ClassNotFoundException e) {
            e.printStackTrace();
            return null;
        }finally {
            try {
                bos.close();
                oos.close();
                bis.close();
                ois.close();
            }catch (Exception e){
                e.printStackTrace();
            }
        }
    }

}
public class Test {

    public static void main(String args[]) throws IOException {

        Address addr = new Address();
        addr.setAdd("杭州市");
        Student stu1 = new Student();
        stu1.setNumber(123);
        stu1.setAddr(addr);

        //调用序列化克隆对象的方法
        Student stu2=(Student) stu1.deepClone();

        System.out.println("学生1:" + stu1.getNumber() + ",地址:" + stu1.getAddr().getAdd());
        System.out.println("学生2:" + stu2.getNumber() + ",地址:" + stu2.getAddr().getAdd());

        addr.setAdd("西湖区");

        System.out.println("学生1:" + stu1.getNumber() + ",地址:" + stu1.getAddr().getAdd());
        System.out.println("学生2:" + stu2.getNumber() + ",地址:" + stu2.getAddr().getAdd());
    }
}

与实现Cloneable的clone方法不一样,这里是实现序列化接口Serializable,然后对需要克隆的对象进行序列化,然后再反序列化对这个对象进行克隆。对应的代码如下;

//对象序列化实现深拷贝
public Object deepClone() throws IOException {
    //创建流对象
    ByteArrayOutputStream bos=null;
    ObjectOutputStream oos=null;
    ByteArrayInputStream bis=null;
    ObjectInputStream ois=null;

    try{
        //序列化
        bos=new ByteArrayOutputStream();
        //字节数组输出流->对象输出流
        oos=new ObjectOutputStream(bos);
        //将当前对象以对象流的方式输出,如果对象中包含对象的引用,那么也会序列化
        oos.writeObject(this);

        //反序列化
        //对象输出流将对象输出后,再序列化读取,相当于进行对象的克隆
        bis=new ByteArrayInputStream(bos.toByteArray());
        ois=new ObjectInputStream(bis);
        //获取克隆对象
        Student student =(Student) ois.readObject();
        return student;
    } catch (ClassNotFoundException e) {
        e.printStackTrace();
        return null;
    }finally {
        try {
            bos.close();
            oos.close();
            bis.close();
            ois.close();
        }catch (Exception e){
            e.printStackTrace();
        }
    }
}

那么就有两种方法实现深拷贝:

1、通过重写clone方法来实现深拷贝

2、通过实现对象序列化来实现深拷贝(推荐)

使用对象序列化进行深拷贝,如果成员变量有对象,可以不需要像clone()方法那样

stu.addr = (Address)addr.clone();	//深度复制

如果类的成员变量很多都是对象类型,那么这个赋值就很繁琐,通过实现对象序列化就不需要考虑成员变量是对象类型的个数了

  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值