Java对象的复制三种方式(小结)

 更新时间:2019年08月27日 10:22:39   作者:chun_soft  
这篇文章主要介绍了Java对象的复制三种方式,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧

1、概述

在实际编程过程中,我们常常要遇到这种情况:有一个对象A,在某一时刻A中已经包含了一些有效值,此时可能 会需要一个和A完全相同新对象B,并且此后对B任何改动都不会影响到A中的值,也就是说,A与B是两个独立的对象,但B的初始值是由A对象确定的。例如下面程序展示的情况:

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.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指向内存堆中同一个对象。如图:

那么,怎么能干干净净清清楚楚地复制一个对象呢。在 Java语言中,用简单的赋值语句是不能满足这种需求的。要满足这种需求有很多途径,

(1)将A对象的值分别通过set方法加入B对象中;
(2)通过重写java.lang.Object类中的方法clone();
(3)通过org.apache.commons中的工具类BeanUtils和PropertyUtils进行对象复制;
(4)通过序列化实现对象的复制。

2、将A对象的值分别通过set方法加入B对象中

对属性逐个赋值,本实例为了演示简单就设置了一个属性:

Student stu1 = new Student(); 
stu1.setNumber(12345); 
Student stu2 = new Student(); 
stu2.setNumber(stu1.getNumber());

我们发现,属性少对属性逐个赋值还挺方便,但是属性多时,就需要一直get、set了,非常麻烦。

3、重写java.lang.Object类中的方法clone()

先介绍一下两种不同的克隆方法,浅克隆(ShallowClone)和深克隆(DeepClone)。

在Java语言中,数据类型分为值类型(基本数据类型)和引用类型,值类型包括int、double、byte、boolean、char等简单数据类型,引用类型包括类、接口、数组等复杂类型。浅克隆和深克隆的主要区别在于是否支持引用类型的成员变量的复制,下面将对两者进行详细介绍。

3.1 浅克隆

一般步骤:

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

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

class Student implements Cloneable{ 
  private int number; 
 
  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[]) { 
    Student stu1 = new Student(); 
    stu1.setNumber(12345); 
    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

在浅克隆中,如果原型对象的成员变量是值类型,将复制一份给克隆对象;如果原型对象的成员变量是引用类型,则将引用对象的地址复制一份给克隆对象,也就是说原型对象和克隆对象的成员变量指向相同的内存地址。

简单来说,在浅克隆中,当对象被复制时只复制它本身和其中包含的值类型的成员变量,而引用类型的成员对象并没有复制。


在Java语言中,通过覆盖Object类的clone()方法可以实现浅克隆。

3.2 深克隆

package abc; 
 
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()); 
     
    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,地址:杭州市

在深克隆中,无论原型对象的成员变量是值类型还是引用类型,都将复制一份给克隆对象,深克隆将原型对象的所有引用对象也复制一份给克隆对象。

简单来说,在深克隆中,除了对象本身被复制外,对象所包含的所有成员变量也将复制。


在Java语言中,如果需要实现深克隆,可以通过覆盖Object类的clone()方法实现,也可以通过序列化(Serialization)等方式来实现。

(如果引用类型里面还包含很多引用类型,或者内层引用类型的类里面又包含引用类型,使用clone方法就会很麻烦。这时我们可以用序列化的方式来实现对象的深克隆。)

4、工具类BeanUtils和PropertyUtils进行对象复制

Student stu1 = new Student(); 
stu1.setNumber(12345); 
Student stu2 = new Student(); 
BeanUtils.copyProperties(stu2,stu1);

这种写法无论多少种属性都只需要一行代码搞定,很方便吧!除BeanUtils外还有一个名为PropertyUtils的工具类,它也提供copyProperties()方法,作用与BeanUtils的同名方法十分相似,主要的区别在于BeanUtils提供类型转换功能,即发现两个JavaBean的同名属性为不同类型时,在支持的数据类型范围内进行转换,而PropertyUtils不支持这个功能,但是速度会更快一些。在实际开发中,BeanUtils使用更普遍一点,犯错的风险更低一点。

5、通过序列化实现对象的复制

序列化就是将对象写到流的过程,写到流中的对象是原有对象的一个拷贝,而原对象仍然存在于内存中。通过序列化实现的拷贝不仅可以复制对象本身,而且可以复制其引用的成员对象,因此通过序列化将对象写到一个流中,再从流里将其读出来,可

以实现深克隆。需要注意的是能够实现序列化的对象其类必须实现Serializable接口,否则无法实现序列化操作。

相关文章

  • 实例详解Java调用第三方接口方法

    实例详解Java调用第三方接口方法

    很多项目都会封装规定好本身项目的接口规范,所以大多数需要去调用对方提供的接口或第三方接口(短信、天气等),下面这篇文章主要给大家介绍了关于Java调用第三方接口方法的相关资料,需要的朋友可以参考下
    2022-06-06
  • java和Spring中观察者模式的应用详解

    java和Spring中观察者模式的应用详解

    这篇文章主要介绍了java和Spring中观察者模式的应用,,具有一定的参考价值,感兴趣的可以了解一下,希望能够给你带来帮助
    2021-10-10
  • 详解Java之路(五) 访问权限控制

    详解Java之路(五) 访问权限控制

    本篇文章主要介绍了Java之路(五) 访问权限控制 ,小编觉得挺不错的,现在分享给大家,也给大家做个参考。
    2016-12-12
  • Spring的Aware接口实现及执行顺序详解

    Spring的Aware接口实现及执行顺序详解

    这篇文章主要为大家介绍了Spring的Aware接口实现及执行顺序详解,有需要的朋友可以借鉴参考下,希望能够有所帮助,祝大家多多进步,早日升职加薪
    2022-12-12
  • 浅谈Java8 判空新写法

    浅谈Java8 判空新写法

    在开发过程中很多时候会遇到判空校验,如果不做判空校验则会产生NullPointerException异常,本文就来介绍一下Java8 判空新写法,感兴趣的可以了解一下
    2021-09-09
  • mybatis的动态SQL和模糊查询实例详解

    mybatis的动态SQL和模糊查询实例详解

    这篇文章主要给大家介绍了关于mybatis的动态SQL和模糊查询的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-03-03
  • Spring中事务用法示例及实现原理详解

    Spring中事务用法示例及实现原理详解

    这篇文章主要给大家介绍了关于Spring中事务用法示例及实现原理的相关资料,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2018-11-11
  • 基于springboot 配置文件context-path的坑

    基于springboot 配置文件context-path的坑

    这篇文章主要介绍了基于springboot 配置文件context-path的坑,基于很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-01-01
  • Java中的程序计数器是什么

    Java中的程序计数器是什么

    这篇文章主要介绍了Java中的程序计数器是什么,帮助大家更好的理解和学习Java,感兴趣的朋友可以了解下
    2020-09-09
  • Java解析word,获取文档中图片位置的方法

    Java解析word,获取文档中图片位置的方法

    下面小编就为大家分享一篇Java解析word,获取文档中图片位置的方法,具有很好的参考价值,希望对大家有所帮助。一起跟随小编过来看看吧
    2018-01-01

最新评论