Java实现克隆的三种方式实例总结

本文实例讲述了Java实现克隆的三种方式。分享给大家供大家参考,具体如下:

1、浅复制(浅克隆)这种浅复制,其实也就是把被复制的这个对象的一些变量值拿过来了。最后生成student2还是一个新的对象。

public class CloneTest1
{
  public static void main(String[] args) throws Exception
  {
    Student1 student = new Student1();
    student.setAge(24);
    student.setName("niesong");
    Student1 student2 = (Student1)student.clone();
    //这个是调用下面的那个方法,然后把这个这个对象Clone到student
    System.out.println("Age:" + student2.getAge() + " " + "Name:" + student2.getName());
    System.out.println("---------------------");
    student2.setAge(23);
    //克隆后得到的是一个新的对象,所以重新写的是student2这个对象的值
    System.out.println(student.getAge());
    System.out.println(student2.getAge());
  }
}
//克隆的对象必须实现Cloneable这个接口,而且需要重写clone方法
class Student1 implements Cloneable
{
  private int age;
  //定义为private说明这个成员变量只能被被当前类中访问,如果外部需要获得,那么就只能通过getAge方法进行获取
  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;
  }
  @Override
  public Object clone() throws CloneNotSupportedException
  {
    Object object = super.clone();
    return object;
  }
}

运行结果:

2、深复制(情况1使用的是在克隆的时候手动进行深克隆)

public class CloneTest2
{
  public static void main(String[] args) throws Exception
  {
    Teacher teacher = new Teacher();
    teacher.setAge(40);
    teacher.setName("teacher zhang");
    Student2 student2 = new Student2();
    student2.setAge(14);
    student2.setName("lisi");
    student2.setTeacher(teacher);
    Student2 student3 = (Student2)student2.clone();
    //这里是深复制,所以这时候Student2中的teacher就是teacher这个对象的一个复制,就和student3是student2的一个复制
    //所以下面teacher.setName只是对他原来的这个对象更改,但是复制的那个并没有更改
    System.out.println(student3.getAge());
    System.out.println(student3.getName());
    System.out.println(student3.getTeacher().getAge());
    teacher.setName("teacher niesong");//不会又任何影响
    System.out.println(student3.getTeacher().getName());
  }
}
class Student2 implements Cloneable
{
  private int age;
  private String name;
  private 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;
  }
  public Teacher getTeacher()
  {
    return teacher;
  }
  public void setTeacher(Teacher teacher)
  {
    this.teacher = teacher;
  }
  @Override
  public Object clone() throws CloneNotSupportedException
  {
    //这一步返回的这个student2还只是一个浅克隆,
    Student2 student2 = (Student2)super.clone();
    //然后克隆的过程中获得这个克隆的student2,然后调用这个getTeacher这个方方法得到这个Teacher对象。然后实现克隆。在设置到这个student2中的Teacher。
    //这样实现了双层克隆使得那个teacher对象也得到了复制。
    student2.setTeacher((Teacher)student2.getTeacher().clone());
    //双层克隆使得那个teacher对象也得到了复制
    return student2;
  }
}
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;
  }
  @Override
  public Object clone() throws CloneNotSupportedException
  {
    return super.clone();
  }
}

运行结果:

3、利用serializable实现深复制(这个是利用Serializable,利用序列化的方式来实现深复制(深克隆),在其中利用了Io流的方式将这个对象写到IO流里面,然后在从IO流里面读取,这样就实现了一个复制,然后实现序列化的这个会将引用的那个对象也一并进行深复制,这样就实现了这个机制,同时在IO里面读取数据的时候还使用了装饰者模式)

import java.io.ByteArrayInputStream;
import java.io.ByteArrayOutputStream;
import java.io.ObjectInputStream;
import java.io.ObjectOutputStream;
import java.io.Serializable;
public class CloneTest3
{
  public static void main(String[] args) throws Exception
  {
    Teacher3 teacher3 = new Teacher3();
    teacher3.setAge(23);
    teacher3.setName("niesong");
    Student3 student3 = new Student3();
    student3.setAge(50);
    student3.setName("wutao");
    student3.setTeacher3(teacher3);
    Student3 ss = (Student3)student3.deepCopt();
    System.out.println(ss.getAge());
    System.out.println(ss.getName());
    System.out.println("---------------------");
    System.out.println(ss.getTeacher3().getAge());
    System.out.println(ss.getTeacher3().getName());
    System.out.println("-----------------------");
    ss.getTeacher3().setAge(7777);
    ss.getTeacher3().setName("hhhhh");
    System.out.println(teacher3.getAge());
    System.out.println(teacher3.getName());
    //虽然上面的已经改了,但是改的是那个复制对象后的那个里面的,然后那个原来的那个里面的并没有改,下面验证:::
    System.out.println("-----------------");
    System.out.println(ss.getTeacher3().getAge());
    System.out.println(ss.getTeacher3().getName());
  }
}
class Teacher3 implements Serializable
{
// 上面的那个警告可以直接消除,除了使用在设置中不显示这个警告,还可以使用下面的这两条语句中的任何一条语句
//  这个serialVersionUID为了让该类别Serializable向后兼容
//  private static final long serialVersionUID = 1L;
//  private static final long serialVersionUID = 8940196742313994740L;
  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 Student3 implements Serializable
{
  private static final long serialVersionUID = 1L;
  private int age;
  private String name;
  private Teacher3 teacher3;
  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 Teacher3 getTeacher3()
  {
    return teacher3;
  }
  public void setTeacher3(Teacher3 teacher3)
  {
    this.teacher3 = teacher3;
  }
  //使得序列化student3的时候也会将teacher序列化
  public Object deepCopt()throws Exception
  {
    ByteArrayOutputStream bos = new ByteArrayOutputStream();
    ObjectOutputStream oos = new ObjectOutputStream(bos);
    oos.writeObject(this);
    //将当前这个对象写到一个输出流当中,,因为这个对象的类实现了Serializable这个接口,所以在这个类中
    //有一个引用,这个引用如果实现了序列化,那么这个也会写到这个输出流当中
    ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray());
    ObjectInputStream ois = new ObjectInputStream(bis);
    return ois.readObject();
    //这个就是将流中的东西读出类,读到一个对象流当中,这样就可以返回这两个对象的东西,实现深克隆
  }
}

运行结果:

更多java相关内容感兴趣的读者可查看本站专题:《Java面向对象程序设计入门与进阶教程》、《Java数据结构与算法教程》、《Java操作DOM节点技巧总结》、《Java文件与目录操作技巧汇总》和《Java缓存操作技巧汇总》

希望本文所述对大家java程序设计有所帮助。

(0)

相关推荐

  • Java编程实现对象克隆(复制)代码详解

    克隆,想必大家都有耳闻,世界上第一只克隆羊多莉就是利用细胞核移植技术将哺乳动物的成年体细胞培育出新个体,甚为神奇.其实在Java中也存在克隆的概念,即实现对象的复制. 本文将尝试介绍一些关于Java中的克隆和一些深入的问题,希望可以帮助大家更好地了解克隆. 假如说你想复制一个简单变量.很简单: int apples = 5; int pears = apples; 不仅仅是int类型,其它七种原始数据类型(boolean,char,byte,short,float,double.long)同样适

  • Java中对象的深复制(深克隆)和浅复制(浅克隆)介绍

    1.浅复制与深复制概念 ⑴浅复制(浅克隆) 被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象.换言之,浅复制仅仅复制所考虑的对象,而不复制它所引用的对象. ⑵深复制(深克隆) 被复制对象的所有变量都含有与原来的对象相同的值,除去那些引用其他对象的变量.那些引用其他对象的变量将指向被复制过的新对象,而不再是原有的那些被引用的对象.换言之,深复制把要复制的对象所引用的对象都复制了一遍. 2.Java的clone()方法 ⑴clone方法将对象复制了一份并返回

  • 基于序列化存取实现java对象深度克隆的方法详解

    我们知道,在java中,将一个非原型类型类型的对象引用,赋值给另一个对象的引用之后,这两个引用就指向了同一个对象,如: 复制代码 代码如下: public class DeepCloneTest { private class CloneTest {  private Long myLong = new Long(1); } public static void main(String args[]) {  new DeepCloneTest().Test(); } public void Te

  • 深入理解Java中的克隆

    前言 Java克隆(Clone)是Java语言的特性之一,但在实际中应用比较少见.但有时候用克隆会更方便更有效率. 对于克隆(Clone),Java有一些限制: 1.被克隆的类必须自己实现Cloneable 接口,以指示 Object.clone() 方法可以合法地对该类实例进行按字段复制.Cloneable 接口实际上是个标识接口,没有任何接口方法. 2.实现Cloneable接口的类应该使用公共方法重写 Object.clone(它是受保护的).某个对象实现了此接口就克隆它是不可能的.即使

  • Java中对象的序列化方式克隆详解

    Java 序列化技术可以使你将一个对象的状态写入一个Byte 流里,并且可以从其它地方把该Byte 流里的数据读出来,重新构造一个相同的对象. 简述: 用字节流的方式,复制Java对象 代码: 流克隆复制函数 public static Object deepClone(Object obj){ if(obj == null){ return null; } try { ByteArrayOutputStream byteOut = new ByteArrayOutputStream(); Ob

  • 解析JAVA深度克隆与浅度克隆的区别详解

    在JAVA克隆对象不能简单的使用clone方法,clone方法只是进行浅克隆.请看下方:深度克隆类:Java代码 复制代码 代码如下: import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; public cla

  • java 对象的克隆(浅克隆和深克隆)

    java 对象的克隆 一.对象的浅克隆 (1)需要克隆类需要重写Object类的clone方法,并且实现Cloneable接口(标识接口,无需实现任何方法) (2)当需要克隆的对象中维护着另外一个引用对象,浅克隆不会克隆另外一个引用对下,而是直接复制维护的另外一个引用对象的地址. (3)对象的浅克隆也不会调用到构造方法. 以下为对象的浅克隆的一个例子: package com.clone; import java.io.Serializable; /** * Description: * 实现了

  • 深入JAVA对象深度克隆的详解

    有时候,我们需要把对象A的所有值复制给对象B(B = A),但是这样用等号给赋值你会发现,当B中的某个对象值改变时,同时也会修改到A中相应对象的值!也许你会说,用clone()不就行了?!你的想法只对了一半,因为用clone()时,除了基础数据和String类型的不受影响外,其他复杂类型(如集合.对象等)还是会受到影响的!除非你对每个对象里的复杂类型又进行了clone(),但是如果一个对象的层次非常深,那么clone()起来非常复杂,还有可能出现遗漏!既然用等号和clone()复制对象都会对原来

  • 浅谈Java中的克隆close()和赋值引用的区别

    学生类Student: package 克隆clone; /*要克隆必须实现这个借口:Cloneable,以标记这个对象可以克隆 Cloneable:此类实现了 Cloneable 接口,以指示 Object.clone() 方法可以合法地对该类实例进行按字段复制. 这个接口是标记接口,告诉我们实现该接口的类就可以实现对象的复制了. */ public class Student implements Cloneable { private String name; private int ag

  • 浅析Java中clone()方法浅克隆与深度克隆

    现在Clone已经不是一个新鲜词语了,伴随着"多莉"的产生这个词语确实很"火"过一阵子,在Java中也有这么一个概念,它可以让我们很方便的"制造"出一个对象的副本来,下面来具体看看Java中的Clone机制是如何工作的?      1. Clone&Copy 假设现在有一个Employee对象,Employee tobby =new Employee("CMTobby",5000),通 常我们会有这样的赋值Employ

随机推荐