Java实现克隆的三种方式实例总结
作者:疯狂1024 发布时间:2021-11-21 15:26:14
标签: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程序设计有所帮助。
来源:https://blog.csdn.net/qq_28081081/article/details/80455150
0
投稿
猜你喜欢
- 如下所示:import java.security.MessageDigest;import java.security.NoSuchAlg
- 一、Thread 类了解如何使用Thread 类实现多线程之后,继续学习Thread 类实现多线程之后的相关功能及方法。1、操作线程名称的方
- 一、简介JetCache是一个基于Java的缓存系统封装,提供统一的API和注解来简化缓存的使用。 JetCache提供了比SpringCa
- 开篇我们还是和原来一样,讲一讲做爬虫的思路以及需要准备的知识吧,高手们请直接忽略。首先我们来缕一缕思绪,想想到底要做什么,列个简单的需求。需
- 1 运算符1.1 概述运算符 用于连接 表达式 的 操作数,并对操作数执行运算。例如,表达式num1+num2,其操作数是num1和num2
- 网络编程是指编写运行在多个设备(计算机)的程序,这些设备都通过网络连接起来。java.net包中J2SE的API包含有类和接口,它们提供低层
- 1.使用的是maven项目,添加依赖<!-- mybatis-plus begin --> <depend
- 一、why(为什么要用Hibernate缓存?)Hibernate是一个持久层框架,经常访问物理数据库。为了降低应用程序对物理数据源访问的频
- PreparedStatement介绍可以通过调用 Connection 对象的 prepareStatement(String sql)
- 项目数据库中出现许多值为中括号[]的数据,测试报了bug,经过排查是因为使用了json-lib 的jar包导致。json-lib在将xml字
- 前言注解(Annotation)不是程序,但可以对程序作出解释,也可以被其它程序(如编译器)读取。注解的格式:以@注释名在代码中存在,还可以
- 前两天朋友问我,有没有使用过StackExchange.Redis,问我要个封装类,由于之前都是使用ServiceStack.Redis,由
- 本文章向大家讲解java中时间的获取和格式化, 一. 获取当前系统时间和日期并格式化输出:import java.util.Dat
- Command模式是最让我疑惑的一个模式,我在阅读了很多代码后,才感觉隐约掌握其大概原理,我认为理解设计模式最主要是掌握起原理构造,这样才对
- 由于一个线程的程序,如果调用一个功能是阻塞的,那么就会影响到界面的更新,导致使用人员操作不便。所以往往会引入双线程的工作的方式,主线程负责更
- java对象拷贝详解及实例Java赋值是复制对象引用,如果我们想要得到一个对象的副本,使用赋值操作是无法达到目的的:@Testpublic
- Eureka大白话解析笔记补录:1.Eureka 介绍Spring Cloud Eureka 是 Spring Clou
- C# goto 语句用于直接在一个程序中转到程序中的标签指定的位置,标签实际上由标识符加上冒号构成。语法形式如下。goto Labell;
- 这篇文章主要介绍了Java实现发送手机短信语音验证功能代码实例,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,
- springboot集成开发实现商场秒杀加入主要依赖<dependency> <groupId>org.spring