时间:2021-05-19
本文实例讲述了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程序设计有所帮助。
声明:本页内容来源网络,仅供用户参考;我单位不保证亦不表示资料全面及准确无误,也不保证亦不表示这些资料为最新信息,如因任何原因,本网内容或者用户因倚赖本网内容造成任何损失或损害,我单位将不会负任何法律责任。如涉及版权问题,请提交至online#300.cn邮箱联系删除。
Java多线程实例3种实现方法Java中的多线程有三种实现方式:1.继承Thread类,重写run方法。Thread本质上也是一个实现了Runnable的实例,
本文实例为大家分享了Java实现多线程的三种方式,供大家参考,具体内容如下importjava.util.concurrent.Callable;importj
iOS应用内部实现AppStore评分功能,笔着整理总结有三种方式,各位可根据自己需求自己选择。先介绍下评分功能实现的三种方式。1,通用方式通过App内部打开网
本文实例讲述了Java实现指定线程执行顺序的三种方式。分享给大家供大家参考,具体如下:方法一:通过共享对象锁加上可见变量来实现。publicclassMySer
spring实现定时任务的方式有三种,分别是java自带的timer类、springtask和quartz三种。 本文只介绍spring自带的task和第三方