Part6:Java中的克隆


@


Example01:Java对象的假克隆

  • 对象的克隆是Java中的一项高级技术,获得与其相同的对象。

  基本数据类型可以使用“=”来进行克隆,此时两个变量除了相等是没有任何关系的。而对于引用类型数据不能简单地使用“=”进行克隆,这与Java的内存空间使用有关。   

  Java将内存空间分成两块,即栈和堆。在栈中保存基本类型和引用变量;在堆中保存对象。对于引用变量而言,使用“=”将修改引用,而不是复制堆中的对象。此时两个引用变量将指向同一个对象。因此,如果一个变量对其修改则会改变另一个变量。

运行结果:



代码实现:

public class Employee {
private String name;
private int age; //省略set()和get()方法 @Override
public String toString() {
return "Employee{" +
"name='" + name + '\'' +
", age=" + age +
'}';
} public static void main(String[] args) {
System.out.println("-----克隆之前:--------");
Employee employee1 = new Employee();
employee1.setName("hyn");
employee1.setAge(20);
System.out.println("员工1的信息:\n"+employee1); System.out.println("-----克隆之后:--------");
Employee employee2 = employee1; //将employee1赋值给employee2
employee2.setName("azw");
employee2.setAge(21);
System.out.println("员工1的信息:\n"+employee1);
System.out.println("员工2的信息:\n"+employee2);
}
}

Example02:Java对象的浅克隆

  在克隆对象时,如果对象的成员变量是基本数据类型,则使用浅克隆即可完成。如果对象的成员变量包括可变引用类型,则需要深克隆。

运行结果:



代码实现:

//Address.java
public class Address {
private String state; //所在国家
private String province; //所在省
private String city; //所在城市 public Address(String state, String province, String city) {
this.state = state;
this.province = province;
this.city = city;
} //省略set()和get()方法 @Override
public String toString() {
StringBuilder sb = new StringBuilder();
sb.append("国家:"+state+",");
sb.append("省:"+province+",");
sb.append("市:"+city);
return sb.toString();
}
}
//Employee.java
public class Employee implements Cloneable{
private String name;
private int age;
private Address address; public Employee(String name, int age, Address address) {
this.name = name;
this.age = age;
this.address = address;
}
//省略set()和get()方法 @Override
public String toString() {
StringBuilder sb = new StringBuilder();
sb.append("姓名:"+name+",");
sb.append("年龄:"+age+",");
sb.append("\n地址:"+address);
return sb.toString();
} @Override
public Employee clone() throws CloneNotSupportedException { //实现浅克隆
Employee employee = (Employee) super.clone();
return employee;
}
}

测试代码:

class Test {
public static void main(String[] args) throws CloneNotSupportedException {
System.out.println("*****克隆之前:******");
Address address = new Address("中国", "湖北", "武汉");
Employee employee1 = new Employee("azw", 20, address);
System.out.println("员工1的信息:\n" + employee1); //employee1的信息 System.out.println("*****克隆之后:******");
Employee employee2 = employee1.clone(); //使用克隆创建Employee2
employee2.getAddress().setState("中国"); //修改地址
employee2.getAddress().setProvince("黑龙江");
employee2.getAddress().setCity("哈尔滨");
employee2.setName("hyn");
employee2.setAge(21);
System.out.println("员工1的信息:\n" + employee1); System.out.println("员工2的信息:\n" + employee2);
}
}
  • 如果引用类型是不可变的,如String类对象,则不必进行深克隆。

Example03:Java对象的深克隆

  • 如果类的成员变量中包括可变引用类型,则需进行深克隆。

运行结果:



代码实现:

//Address.java
public class Address implements Cloneable{
private String state; //所在国家
private String province; //所在省
private String city; //所在城市 public Address(String state, String province, String city) {
this.state = state;
this.province = province;
this.city = city;
} //省略set()和get()方法 @Override
public String toString() {
StringBuilder sb = new StringBuilder();
sb.append("国家:"+state+",");
sb.append("省:"+province+",");
sb.append("市:"+city);
return sb.toString();
}
//---------------------------
@Override
public Address clone() throws CloneNotSupportedException {
//Address类中的域不是基本类型就是不可变类型,所以可以直接使用浅克隆
Address address = (Address) super.clone();
return address;
}
//---------------------------
} //Employee.java
public class Employee implements Cloneable{
private String name;
private int age;
private Address address; public Employee(String name, int age, Address address) {
this.name = name;
this.age = age;
this.address = address;
} //省略set()和get()方法 @Override
public String toString() {
StringBuilder sb = new StringBuilder();
sb.append("姓名:"+name+",");
sb.append("年龄:"+age+",");
sb.append("\n地址:"+address);
return sb.toString();
} @Override
public Employee clone() throws CloneNotSupportedException { //实现深克隆
Employee employee = (Employee) super.clone();
//---------------------------------
employee.address = address.clone();
//---------------------------------
return employee;
}
} //测试代码同Example02测试代码.
  • 要点:通常情况下,需要用到克隆对象时都需要使用深克隆。

Example04:序列化与对象克隆

  如果类的成员变量比较复杂,例如使用了多个可变的引用类型,使用clone()方法是非常麻烦的,所以可以考虑序列化的方式完成克隆。

运行结果:



代码实现:

import java.io.Serializable;

public class Employee implements Serializable {
//同Example04中Employee.java的代码
} public class Address implements Serializable {
//同Example04中Assress.java的代码
}

测试代码:

class Test {
public static void main(String[] args) throws IOException, ClassNotFoundException {
System.out.println("*****序列化之前:******");
Address address = new Address("中国", "湖北", "武汉");
Employee employee1 = new Employee("azw", 20, address);
System.out.println("员工1的信息:\n" + employee1); //employee1的信息 System.out.println("*****序列化之后:******");
Employee employee2 = null; ObjectOutputStream out = new ObjectOutputStream(new FileOutputStream("E:\\employee.txt"));
out.writeObject(employee1); //将对象写入到本地文件中 ObjectInputStream in = new ObjectInputStream(new FileInputStream("E:\\employee.txt"));
employee2 = (Employee)in.readObject(); //从本地文件中读取对象 if (employee2 != null) {
employee2.getAddress().setState("中国"); //修改地址
employee2.getAddress().setProvince("黑龙江");
employee2.getAddress().setCity("哈尔滨");
employee2.setName("hyn");
employee2.setAge(21);
System.out.println("员工1的信息:\n" + employee1);
System.out.println("员工2的信息:\n" + employee2);
}
}
}

要点:进行序列化的类需要实现Serializable接口,该接口中并没有定义任何方法,是一个标识接口。如果类中有可变的引用类型成员变量,则该变量需要实现Serializable接口。本实例采用将对象写入本地文件的方式完成序列化。


Example05:深克隆和序列化的效率比较

  • 通过使用这两种方式克隆100000个对象,并输出花费的时间来比较这两种方法的效率。

运行结果:



代码实现:

import java.io.Serializable;

public class Employee implements Cloneable,Serializable {
private String name;
private int age; public Employee(String name, int age) {
this.name = name;
this.age = age;
} @Override
public String toString() {
StringBuilder sb = new StringBuilder();
sb.append("姓名:"+name+",");
sb.append("年龄:"+age+",");
return sb.toString();
} @Override
public Employee clone() throws CloneNotSupportedException { //使用父类的clone()方法实现深克隆
Employee employee = (Employee) super.clone();
return employee;
}
}
测试代码:
import java.io.*;
import java.util.ArrayList;
import java.util.List; class Test {
public static void main(String[] args) throws IOException, ClassNotFoundException, CloneNotSupportedException {
List<Employee> employees = new ArrayList<Employee>(); //创建列表保存对象
Employee employee = new Employee("azw", 20); //创建对象
long currentTime = System.currentTimeMillis(); //获得当前系统时间
//使用克隆方式获得对象
for (int i = 0;i<100000;i++){
employees.add(employee.clone());
}
System.out.println("克隆花费的时间:"+(System.currentTimeMillis()-currentTime)+"毫秒"); currentTime = System.currentTimeMillis(); //获得当前系统时间
for (int i = 0;i<100000;i++){
ByteArrayOutputStream bout = new ByteArrayOutputStream(); //创建字节数组输出流
ObjectOutputStream out = new ObjectOutputStream(bout); //创建对象输出流
out.writeObject(employee); //将对象写入到输出流中
//获得字节输出流内容
ByteArrayInputStream bin = new ByteArrayInputStream(bout.toByteArray());
ObjectInputStream in = new ObjectInputStream(bin); //创建对象输入流
employees.add((Employee) in.readObject()); //读取对象
}
System.out.println("序列化花费的时间:"+(System.currentTimeMillis()-currentTime)+"毫秒");
}
}

要点:使用ByteArrayOutputStream和ByteArrayInputStream可以将对象保存在内存中,这样就不必产生一个本地文件来完成序列化的功能。


假克隆、浅克隆和深克隆的应用范围

假克隆 基本数据类型
浅克隆 基本数据类型、不可变引用类型
深克隆 可变引用类型

Java实例 Part6:Java中的克隆的更多相关文章

  1. Java-Runoob-高级教程-实例-数组:14. Java 实例 – 在数组中查找指定元素

    ylbtech-Java-Runoob-高级教程-实例-数组:14. Java 实例 – 在数组中查找指定元素 1.返回顶部 1. Java 实例 - 在数组中查找指定元素  Java 实例 以下实例 ...

  2. Java-Runoob-高级教程-实例-数组:10. Java 实例 – 查找数组中的重复元素-un

    ylbtech-Java-Runoob-高级教程-实例-数组:10. Java 实例 – 查找数组中的重复元素 1.返回顶部 1. Java 实例 - 查找数组中的重复元素  Java 实例 以下实例 ...

  3. Java-Runoob-高级教程-实例-字符串:03. Java 实例 - 删除字符串中的一个字符

    ylbtech-Java-Runoob-高级教程-实例-字符串:03. Java 实例 - 删除字符串中的一个字符 1.返回顶部 1. Java 实例 - 删除字符串中的一个字符  Java 实例 以 ...

  4. Java实例练习——java实现自动生成长度为10以内的随机字符串(可用于生成随机密码)

    package sorttest; import java.util.ArrayList; import java.util.Collections; import java.util.List; i ...

  5. Java-Runoob-高级教程-实例-方法:06. Java 实例 – 方法覆盖

    ylbtech-Java-Runoob-高级教程-实例-方法:06. Java 实例 – 方法覆盖 1.返回顶部 1. Java 实例 - 方法覆盖  Java 实例 前面章节中我们已经学习了 Jav ...

  6. Java - 35 Java 实例

    Java 实例 本章节我们将为大家介绍 Java 常用的实例,通过实例学习我们可以更快的掌握 Java 的应用. Java 环境设置实例 Java 实例 – 如何编译一个Java 文件? Java 实 ...

  7. Java-Runoob-高级教程:Java 实例

    ylbtech-Java-Runoob-高级教程:Java 实例 1.返回顶部 1. Java 实例 本章节我们将为大家介绍 Java 常用的实例,通过实例学习我们可以更快的掌握 Java 的应用. ...

  8. Java 实例

    Java 实例 本章节我们将为大家介绍 Java 常用的实例,通过实例学习我们可以更快的掌握 Java 的应用. Java 环境设置实例 Java 实例 – 如何编译一个Java 文件? Java 实 ...

  9. Java 学习(21):Java 实例

    Java 实例 本章节我们将为大家介绍 Java 常用的实例,通过实例学习我们可以更快的掌握 Java 的应用. Java 环境设置实例 //HelloWorld.java 文件 public cla ...

随机推荐

  1. 为什么懂云的IT高手能过得比你好

    盼望着,盼望着,一年一度的国庆7天长假还有不到24小时就到来了.各个部门的同事都已准备好满世界旅行去了. IT 部门各位同事的心还是悬着,信息系统还要持续的运转,对外的网站不能停,假期的线上促销也不能 ...

  2. Azure School 终于上线了,物联网开发课程先走一波~

    要说目前最热门的技术是什么,物联网(IoT)肯定荣登榜首!老牌科技企业早已涉足,新晋初创公司层出不穷,就连很多传统企业也开始试水这一领域…… 物联网,说小其实挺小,工业仪表.汽车飞机,甚至家用电器,专 ...

  3. The associated COM server does not support ActiveX Document embedding

    winfrom 通过dsoframer实现读取excel文件报错: System.Reflection.TargetInvocationException: Exception has been th ...

  4. 配置Sharepoint之后。外网无法访问的问题

    Sharepoint配置完成了,projectserver也已经配置完毕.突然遇到一个情况就是外网访问不了,这可麻烦了,费了半天事访问不了等于0啊.没办法,研究吧.在群里问了大神,终于解决了.现将解决 ...

  5. 安装SCOM Reporting Server

    在SQL群集计算机上可以安装SCOM Reporting Server. 1.运行SQL Server安装程序,选择 全新安装SQL Server,不能向已有的群集实例中添加Reporting Ser ...

  6. C# 表达式树 创建、生成、使用、lambda转成表达式树~表达式树的知识详解

    笔者最近学了表达式树这一部分内容,为了加深理解,写文章巩固知识,如有错误,请评论指出~ 表达式树的概念 表达式树的创建有 Lambda法 和 组装法. 学习表达式树需要 委托.Lambda.Func& ...

  7. Samba文件共享服务器配置

    Samba起源: 早期网络想要在不同主机之间共享文件大多要用FTP协议来传输,但FTP协议仅能做到传输文件却不能直接修改对方主机的资料数据,这样确实不太方便,于是便出现了NFS开源文件共享程序:NFS ...

  8. 关于memcpy的实现

    今天去面试,面试官出了一个关于memcpy的函数原型的实现的问题,本来这个问题是很简单的,但是不知道当时怎么脑子一抽竟然写错了,真是”累觉不爱”了.感觉这份工作算是泡汤了,算了事情发生了,错过了也就错 ...

  9. 高性能 Socket 组件 HP-Socket v3.2.1-RC3 公布

    HP-Socket 是一套通用的高性能 TCP/UDP Socket 组件,包括服务端组件.client组件和 Agent 组件,广泛适用于各种不同应用场景的 TCP/UDP 通信系统.提供 C/C+ ...

  10. tar 打包带软连接的文件

    打包普通文件夹,压缩带参数z,创建tar.gz tar -cvf ./tmp/SK_Aug_camera.tar ./gap_40_5 但是文件夹里含有软连接,带参数 h tar -cvhf ./tm ...