Java对象拷贝之引用拷贝,浅拷贝和深拷贝以及常用工具类

对象拷贝分为引用拷贝、浅拷贝和深拷贝三种方式。

1、引用拷贝

引用拷贝只是复制了原对象的引用,即两个对象指向同一块内存堆地址。修改其中的一个对象会影响到另一个对象。

代码举例:

Dog dog1=new Dog("小黄",2);
Dog dog2=dog1;//它们指向的是同一个堆中的地址
System.out.println(dog1);
System.out.println(dog2);
dog2.setName("小黑");//如果我改了其中一个对象的属性值,其他对象的属性会跟着变
System.out.println(dog1.getName());
System.out.println(dog2.getName());

输出结果:

Dog@3d075dc0
Dog@3d075dc0
小黑
小黑

1.1基本类型和引用类型

先来了解一下:

在 Java 中数据类型可以分为两大类:基本类型和引用类型。

基本类型也称为值类型,分别是字符类型 char,布尔类型 boolean以及数值类型 byte、short、int、long、float、double。

引用类型则包括类、接口、数组、枚举、字符串等。

Java 将内存空间分为堆和栈。基本类型直接在栈中存储数值,而引用类型是将引用放在栈中,实际存储的值是放在堆中,通过栈中的引用指向堆中存放的数据。

2、浅拷贝

创建一个新对象,这个对象有着原始对象属性值的一份精确拷贝。如果属性是基本类型,拷贝的就是基本类型的值,如果属性是引用类型,拷贝的就是内存地址,原始对象及其副本引用同一个对象。

实现步骤:

1.实现Cloneable接口,标识这个类对象是可以被拷贝的。

2.重写Object继承而来的clone()方法。在clone方法里捕获异常。

代码举例

public class CopyTest {
    public static void main(String[] args) {
        Person p1=new Person("张三",2,new Car(120));
        Person p2=new Person();
        System.out.println("p1的name属性的地址值");
        System.out.println(p1.getName().getClass().getName()+'@'+Integer.toHexString(System.identityHashCode(p1.getName())));
        p2=p1.clone();//浅拷贝
        p2.setAge(20);
        System.out.println("p2的name属性的地址值");
        System.out.println(p2.getName().getClass().getName()+'@'+Integer.toHexString(System.identityHashCode(p2.getName())));
        p2.setName("王五");
        System.out.println("改了之后p2的name属性的地址值");//注意:浅拷贝里,若变量为String字符串,则拷贝其地址引用。但是在修改时,它会从字符串池中重新生成一个新的字符串,原来的对象保持不变。
        System.out.println(p2.getName().getClass().getName()+'@'+Integer.toHexString(System.identityHashCode(p2.getName())));//这就是为什么它改了值,地址跟原先不一样了。
        System.out.println("p1的car的引用地址"+p1.car);
        p2.car.setMoney(1);//变量是一个实例对象,则拷贝其地址引用,p2和p1公有的
        System.out.println("p2的car的引用地址"+p2.car);
        System.out.println(p1);
        System.out.println(p2);
    }
}
class Person implements Cloneable{
   private String name;
   private   int age;
    Car car;
   //省略对应的构造函数,getter和setter方法
   @Override
    protected Person clone() {
        Person person=null;
        try {
           person= (Person) (super.clone());
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return person;
    }
//省略toString方法
}
class Car{
   private int money;
//省略对应的构造函数,getter和setter方法
}

答案:

注意:浅拷贝里,若变量为String字符串,则拷贝其地址引用。但是在修改时,它会从字符串池中重新生成一个新的字符串,原来的对象保持不变。

3、深拷贝

深拷贝是一种完全拷贝,无论是基本类型还是引用类型都会完完全全的拷贝一份,在内存中生成一个新的对象。简单点说就是拷贝对象和被拷贝对象没有任何关系,互不影响。

在上面浅拷贝那个代码里,显然,我们不希望p2和p1共用一个car对象,可以用下面两种方法去解决。

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

与通过重写clone方法实现浅拷贝的基本思路一样,让每个引用类型属性内部都重写clone() 方法,最后在最顶层的类的重写的c1one方法中调用所有的clone方法即可实现深拷贝。

浅拷贝代码改进

public class BeanUtilsTest {
    public static void main(String[] args) {
        Person p1=new Person("张三",2,new Car(120));
        Person p2=new Person();
        p2=p1.clone();
        p2.setAge(20);
        p2.setName("王五");
        System.out.println("p1的car的引用地址"+p1.car);
        p2.car.setMoney(1);//变量是一个实例对象,则拷贝其地址引用,p2和p1公有的
        System.out.println("p2的car的引用地址"+p2.car);
        System.out.println(p1);
        System.out.println(p2);
    }
}
class Person implements Cloneable{
   private String name;
   private   int age;
    Car car;
//省略对应的构造函数,getter和setter方法
    @Override
    protected Person clone() {
        Person person=null;
        try {
           person= (Person) (super.clone());
           person.car=person.car.clone(); //调用
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return person;
    }
//省略toString方法
}
class Car implements Cloneable{
   private int money;
//省略对应的构造函数,getter和setter方法
    @Override
    protected Car clone(){//在car里重写clone方法
        Car car=null;
        try {
           car=(Car)(super.clone());
        } catch (CloneNotSupportedException e) {
            e.printStackTrace();
        }
        return car;
    }
}

答案:

虽然层次调用clone方法可以实现深拷贝,但是代码量太大,特别是针对属性数量比较多,层次比较深的类而言,每个类都要重写clone方法太过于繁琐。

二、通过对象序列化实现深拷贝

序列化是将对象写到流中便于传输,而反序列化则是把对象从流中读取出来。无论嵌套多少个引用类型,序列化都能实现深拷贝。

注意每个需要序列化的类都要实现 Serializable 接口,如果有某个属性不需要序列化,可以将其声明为 transient,即将其排除在克隆属性之外。

代码举例:

public class BeanUtilsTest {
    public static void main(String[] args) throws IOException, ClassNotFoundException {
        Person p1=new Person("张三",2,new Car(120));
         //序列化
        ByteArrayOutputStream bos=new ByteArrayOutputStream();//字节数组容器
        ObjectOutputStream oos=new ObjectOutputStream(bos);
        oos.writeObject(p1);//把p1转换为字节序列
    //注意,这个序列化和反序列化过程最好写在一个方法里面
        
        //反序列化
        ObjectInputStream ois=new ObjectInputStream(new ByteArrayInputStream(bos.toByteArray()));
        Person p2=(Person)ois.readObject();
        
        p2.setAge(20);
        p2.setName("王五");
        System.out.println("p1的car的引用地址"+p1.car);
        p2.car.setMoney(1);//变量是一个实例对象,则拷贝其地址引用,p2和p1公有的
        System.out.println("p2的car的引用地址"+p2.car);
        System.out.println(p1);
        System.out.println(p2);
    }
}
class Person implements Serializable{
   private String name;
   private   int age;
    Car car;
//省略对应的构造函数,getter和setter方法
//省略toString方法
}
class Car implements Serializable{
   private int money;
   //省略对应的构造函数,getter和setter方法
}

答案:

序列化是把堆内存中的 Java 对象数据,通过某种方式把对象存储到磁盘文件中或者传递给其他网络节点(在网络上传输)。而反序列化则是把磁盘文件中的对象数据或者把网络节点上的对象数据,恢复成Java对象模型的过程。

总结

  • 引用拷贝:当我们把一个对象赋值给一个新的变量时,赋的其实是该对象的在栈中的地址,而不是堆中的数据。也就是两个对象指向的是同一个存储空间,无论哪个对象发生改变,其实都是改变的存储空间的内容。
  • 浅拷贝:重新在堆中创建内存,拷贝前后对象的基本数据类型互不影响,但拷贝前后对象的引用类型因共享同一块内存,会相互影响。
  • 深拷贝:从堆内存中开辟一个新的区域存放新对象,对对象中的子对象进行递归拷贝,拷贝前后的两个对象互不影响。

4、常用拷贝工具类

后续更新....

  • 3
    点赞
  • 4
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
### 回答1: 我可以回答这个问题,以下是一个Java对象深拷贝的实现示例: ```java import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; public class DeepCopyUtil { public static Object deepCopy(Object obj) throws Exception { ByteArrayOutputStream bos = new ByteArrayOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(bos); oos.writeObject(obj); ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray()); ObjectInputStream ois = new ObjectInputStream(bis); return ois.readObject(); } } ``` 使用该工具类,可以对Java对象进行深拷贝。 ### 回答2: Java对象深拷贝工具类主要是用来创建一个完全独立的对象副本,其内部字段的值与原对象完全相同,但是在内存中是独立存在的,对副本的修改不会影响原对象。下面是一个简单的Java对象深拷贝工具类的实现: ```java import java.io.*; public class DeepCopyUtils { public static <T extends Serializable> T deepCopy(T object) { try { // 将对象写入流中 ByteArrayOutputStream baos = new ByteArrayOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(baos); oos.writeObject(object); oos.close(); // 从流中读取对象 ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray()); ObjectInputStream ois = new ObjectInputStream(bais); T copyObject = (T) ois.readObject(); ois.close(); return copyObject; } catch (Exception e) { e.printStackTrace(); return null; } } } ``` 上述工具类中的deepCopy方法接受一个泛型对象作为参数,并返回该对象深拷贝副本。实现过程利用了Java的序列化机制,即将对象写入输出流再从输入流读取出来,从而实现完全独立的副本。 需要注意的是,使用该深拷贝工具类对象必须实现Serializable接口,即可被序列化。另外,在实际应用中需要注意,被拷贝对象引用类型字段也必须实现Serializable接口,否则会抛出NotSerializableException异常。 使用该深拷贝工具类,可以轻松实现Java对象深拷贝,确保在进行对象拷贝的过程中保持原对象拷贝对象的独立性。 ### 回答3: Java对象深拷贝即在拷贝一个对象时,创建一个新的对象,并且将原始对象中的所有属性值都复制到新对象中,而不是共享原始对象的属性值。为了实现Java对象深拷贝,可以使用以下的工具类: 1. 使用Java序列化:可以将对象序列化为字节流,然后再反序列化成一个新的对象。这个过程会创建一个全新的对象,所有属性都是独立的副本。下面是一个示例代码: ``` import java.io.*; public class DeepCopyUtil { public static Object deepCopy(Object object) { try { ByteArrayOutputStream baos = new ByteArrayOutputStream(); ObjectOutputStream oos = new ObjectOutputStream(baos); oos.writeObject(object); ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray()); ObjectInputStream ois = new ObjectInputStream(bais); return ois.readObject(); } catch (Exception e) { e.printStackTrace(); return null; } } } ``` 2. 使用clone()方法:Java中的所有对象都继承了Object类,而Object类中有一个clone()方法,可以实现对象浅拷贝。但是要实现深拷贝,需要在需要拷贝对象中实现Cloneable接口,并重写clone()方法。在clone()方法中,可以对每个属性进行拷贝。下面是一个示例代码: ``` public class DeepCopyUtil { public static Object deepCopy(Object object) { try { if (object instanceof Cloneable) { Method cloneMethod = object.getClass().getDeclaredMethod("clone"); cloneMethod.setAccessible(true); return cloneMethod.invoke(object); } } catch (Exception e) { e.printStackTrace(); } return null; } } ``` 以上是两种常用的实现Java对象深拷贝工具类方式,可以根据实际需求选择适合的方法。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值