java克隆

为什么需要克隆

我们在很多时候需要使用一个对象去记录另外一个对象的当前状态,对象中可能会有很多属性,如果我们一个一个去设置,不仅不方便,而且效率很低,我们看一个初学者可能遇到的问题

class Person{
String name;
int age; public Person() {}
public Person(String name, int age) {
super();
this.name = name;
this.age = age;
}
@Override
public String toString() {
return "Person [name=" + name + ", age=" + age + "]";
} } @Test
public void test2() {
Person p1=new Person("tom",20);
Person p2=p1;
System.out.println(p1);
System.out.println(p2);
System.out.println("---------");
p1.age=30;
System.out.println(p1);
System.out.println(p2);
}
输出:
Person [name=tom, age=20]
Person [name=tom, age=20]
---------
Person [name=tom, age=30]
Person [name=tom, age=30]

也许有的人认为Person p2=p1这样的方式就可以克隆一个对象,这种想法是错误的,这种使用等号赋值的方式只是将p1的地址赋值给了p2对象,那么p1和p2都指向了堆中的同一个对象,所以,修改p1那么p2也就变了

如果一个一个属性的去手动设置不仅麻烦,而且属性可能也有属性,修改起来不容易

public void test2() {
Person p1=new Person("tom",20);
Person p2=new Person();
p2.age=p1.age;
p2.name=p1.name;
}

这里Person的属性层级很简单,修改起来看起来很简单,但是如果Person多了一个Address类型的属性,那么手动修改就必须要去new一个Address并赋值属性,假如属性的嵌套层级深了,就很难了

所以我们可以使用Object提供的clone方法来克隆对象,由于clone方法是用protected关键字修饰的,我们如果想在类外使用就需要重写父类的方法,Object的clone方法是一个native关键字修饰的方法,即调用其他语言的方法,效率很高,值得注意的一点是要克隆对象的类必须实现Cloneable接口,这个接口是一个标记接口,里面并没有定义任何方法,如果没事实现Cloneable接口使用clone方法,会抛出CloneNotSupportedException异常

浅克隆

如果原型对象的属性是值类型,那么复制一份给克隆对象,如果属性是引用类型,那么把属性的引用赋值给克隆对象

class Person implements Cloneable{//必须实现Cloneable接口
String name;
int age; public Person() {}
public Person(String name, int age) {
super();
this.name = name;
this.age = age;
}
@Override
public String toString() {
return "Person [name=" + name + ", age=" + age + "]";
}
@Override
protected Object clone(){
Person p=null;
try {
p=(Person)super.clone();
}catch(CloneNotSupportedException e) {//实现了Cloneable接口这个异常就不可能发生
e.printStackTrace();
}
return p;
} } @Test
public void test2() {
Person p1=new Person("tom",20);
Person p2=(Person)p1.clone();
System.out.println(p1);
System.out.println(p2);
System.out.println("---------");
p1.age=30;
System.out.println(p1);
System.out.println(p2);
}
输出:
Person [name=tom, age=20]
Person [name=tom, age=20]
---------
Person [name=tom, age=30]
Person [name=tom, age=20]

这种Object提供的clone在类中的属性全是值类型的时候不会出现问题,但是如果类属性有引用类型就会出现问题

class Person implements Cloneable{
String name;
int age;
Address address; public Person() {}
public Person(String name, int age,Address addr) {
super();
this.name = name;
this.age = age;
this.address=addr;
} @Override
public String toString() {
return "Person [name=" + name + ", age=" + age + ", address=" + address + "]";
}
@Override
protected Object clone(){
Person p=null;
try {
p=(Person)super.clone();
}catch(CloneNotSupportedException e) {
e.printStackTrace();
}
return p;
} }
class Address{
String addr; public Address(String addr) {
super();
this.addr = addr;
} @Override
public String toString() {
return "Address [addr=" + addr + "]";
} } @Test
public void test2() {
Address addr=new Address("成都市郫都区");
Person p1=new Person("tom",20,addr);
Person p2=(Person)p1.clone();
System.out.println(p1);
System.out.println(p2);
System.out.println("---------");
p1.address.addr="成都市金牛区";
System.out.println(p1);
System.out.println(p2);
}
输出:
Person [name=tom, age=20, address=Address [addr=成都市郫都区]]
Person [name=tom, age=20, address=Address [addr=成都市郫都区]]
---------
Person [name=tom, age=20, address=Address [addr=成都市金牛区]]
Person [name=tom, age=20, address=Address [addr=成都市金牛区]]

修改p1的address根据p1克隆出来的p2的address也改变了,这就是所说的当克隆对象的属性是引用类型时,只会复制它的引用,而这种情况一般是我们不希望的,所以需要使用深克隆

深克隆

1.克隆对象所有的引用类型都重写clone方法

这里所说的引用类型重写clone方法,是指克隆对象本身属性是引用类型的必须重写clone方法且引用类型中的引用类型也必要要重写,且必须在clone方法中显式条用

class Person implements Cloneable{
String name;
int age;
Address address;//这个属性是引用类型,必须实现Cloneable接口重写clone方法 public Person() {}
public Person(String name, int age,Address addr) {
super();
this.name = name;
this.age = age;
this.address=addr;
} @Override
public String toString() {
return "Person [name=" + name + ", age=" + age + ", address=" + address + "]";
}
@Override
protected Object clone(){
Person p=null;
try {
p=(Person)super.clone();
}catch(CloneNotSupportedException e) {
e.printStackTrace();
}
//显式调用克隆引用数据类型
p.address=(Address)address.clone();
return p;
} }
class Address implements Cloneable{
String addr; public Address(String addr) {
super();
this.addr = addr;
} @Override
public String toString() {
return "Address [addr=" + addr + "]";
} @Override
protected Object clone() {
Address addr=null;
try {
addr=(Address)super.clone();
}catch(CloneNotSupportedException e) {
e.printStackTrace();
}
return addr;
} } @Test
public void test2() {
Address addr=new Address("成都市郫都区");
Person p1=new Person("tom",20,addr);
Person p2=(Person)p1.clone();
System.out.println(p1);
System.out.println(p2);
System.out.println("---------");
p1.address.addr="成都市金牛区";
System.out.println(p1);
System.out.println(p2);
}
输出:
Person [name=tom, age=20, address=Address [addr=成都市郫都区]]
Person [name=tom, age=20, address=Address [addr=成都市郫都区]]
---------
Person [name=tom, age=20, address=Address [addr=成都市金牛区]]
Person [name=tom, age=20, address=Address [addr=成都市金牛区]]

使用序列化方法

需要克隆的对象类必须实现Serializable接口,这个接口也是一个标记接口,接口中没有任何方法,这个方法的实现类表示可以将这个类的对象写入到IO流中,那么久可以把对象在网络中发送或保存到本地磁盘

class Person implements Cloneable,Serializable{
/**
*
*/
private static final long serialVersionUID = 8990580911834489134L;
String name;
int age;
Address address; public Person() {}
public Person(String name, int age,Address addr) {
super();
this.name = name;
this.age = age;
this.address=addr;
} @Override
public String toString() {
return "Person [name=" + name + ", age=" + age + ", address=" + address + "]";
}
@Override
protected Object clone(){
Person p=null;
try {
//将对象写入流中
ByteArrayOutputStream baos=new ByteArrayOutputStream();
ObjectOutputStream oos=new ObjectOutputStream(baos);
oos.writeObject(this);
//将对象从流中读取出来
ByteArrayInputStream bais=new ByteArrayInputStream(baos.toByteArray());
ObjectInputStream ois=new ObjectInputStream(bais);
p=(Person)ois.readObject();
} catch (ClassNotFoundException e) {
e.printStackTrace();
} catch (IOException e) {
e.printStackTrace();
}
return p;
} }
class Address implements Serializable{
/**
*
*/
private static final long serialVersionUID = 328854588872604721L;
String addr; public Address(String addr) {
super();
this.addr = addr;
} @Override
public String toString() {
return "Address [addr=" + addr + "]";
}
} @Test
public void test2() {
Address addr=new Address("成都市郫都区");
Person p1=new Person("tom",20,addr);
Person p2=(Person)p1.clone();
System.out.println(p1);
System.out.println(p2);
System.out.println("---------");
p1.address.addr="成都市金牛区"; System.out.println(p1);
System.out.println(p2);
}
输出:
Person [name=tom, age=20, address=Address [addr=成都市郫都区]]
Person [name=tom, age=20, address=Address [addr=成都市郫都区]]
---------
Person [name=tom, age=20, address=Address [addr=成都市金牛区]]
Person [name=tom, age=20, address=Address [addr=成都市郫都区]]

java对象clone的更多相关文章

  1. Java对象clone()的测试

    Object中自带native clone()方法. 研究了一下用法. public class DeepCopyTest { public static void main(String[] arg ...

  2. Java对象克隆(Clone)及Cloneable接口、Serializable接口的深入探讨

    Java对象克隆(Clone)及Cloneable接口.Serializable接口的深入探讨 Part I 没啥好说的,直接开始Part II吧. Part II 谈到了对象的克隆,就不得不说为什么 ...

  3. java对象转json应clone,避免生成json串有问题

    今天因为一个java对象转json,搞了我一下午,在些记录一下: 是这样:我在strtuts2的action中调用services返回 Row: 26, 中国银行海鹰, 29, 东楼, 36, 1F ...

  4. java克隆对象clone()的使用方法和作用

    转自:997.html">http://www.okrs.cn/blog/news/?997.html 内容摘要 若需改动一个对象,同一时候不想改变调用者的对象.就要制作该对象的一个本 ...

  5. Java基础——clone()方法浅析

    一.clone的概念 clone顾名思义就是复制, 在Java语言中, clone方法被对象调用,所以会复制对象.所谓的复制对象,首先要分配一个和源对象同样大小的空间,在这个空间中创建一个新的对象.那 ...

  6. 分析java中clone()方法 (转载+修改)

    Java中的clone() 方法 java所有的类都是从java.lang.Object类继承而来的,而Object类提供下面的方法对对象进行复制. protected native Object c ...

  7. java中clone的深入理解

    Java中Clone的概念大家应该都很熟悉了,它可以让我们很方便的“制造”出一个对象的副本来,下面来具体看看java中的Clone机制是如何工作的?      1. Clone和Copy      假 ...

  8. Java 对象复制

    Java 对象的一共有 3 种复制对象的方式. 1.直接赋值 (引用复制 ),此种复制方式比较常用. 诸如 A  a = b ;  a 是直接复制了b的引用 ,也就是说它俩指向的是同一个对象. 此时 ...

  9. 【JAVA零基础入门系列】Day14 Java对象的克隆

    今天要介绍一个概念,对象的克隆.本篇有一定难度,请先做好心理准备.看不懂的话可以多看两遍,还是不懂的话,可以在下方留言,我会看情况进行修改和补充. 克隆,自然就是将对象重新复制一份,那为什么要用克隆呢 ...

随机推荐

  1. Yuchuan_Linux_C编程之一 Vim编辑器的使用

    一.整体大纲 二.Vim 编辑器的使用 vi -- vim    vim是从vi发展过来的一款文本编辑器    vi a.txt    前提: 安装了vim软件 工作模式: 1. 命令模式 -- 打开 ...

  2. css 居中方法

    垂直居中 利用“精灵元素”(ghost element)技术实现垂直居中,即在父容器内放一个100%高度的伪元素,让文本和伪元素垂直对齐,从而达到垂直居中的目的. .ghost-center { po ...

  3. HDFS DataNode详解

    1. datanode介绍 1.1 datanode datanode是负责当前节点上的数据的管理,具体目录内容是在初始阶段自动创建的,保存的文件夹位置由配置选项{dfs.data.dir}决定 1. ...

  4. IoT设备实践丨如果你也在树莓派上部署了k3s,你也许需要这篇文章

    前 言 树莓派是一种广泛流行的开发板,随着物联网的深入发展,树莓派大有成为IoT终端设备标准之趋势.在支持客户在IoT场景中落地k3s时,k3s在树莓派上的部署问题也就出现了.本文记录了一些其中的关键 ...

  5. git的日常使用(补课)

    使用git的一些反响 如果在github上做一个仓库来使用的话... 首先手动在github上创建一个空的仓库,默认没有任何东西的 使用git的命令 git pull 创建的github仓库地址 能把 ...

  6. element UI使用

    1.Button按钮 <el-button type="text">文字按钮</el-button>设置type="text",可以是无 ...

  7. 欲善事先利器-IEAD插件篇

    工欲善其事,必先利其器,好鞋踢好球是非常合乎逻辑的事情. --<长江七号> 同样的开场白,不一样的酒,不一样的故事. 上篇<欲善事先利器--系统篇>已经推荐了一些个人常用的效率 ...

  8. Multi-batch TMT reveals false positives, batch effects and missing values(解读人:胡丹丹)

    文献名:Multi-batch TMT reveals false positives, batch effects and missing values (多批次TMT定量方法中对假阳性率,批次效应 ...

  9. Python-操作XML文件

    一.python对XML文件的操作 目录 1.xml 创建 2.xml 数据查询 3.xml 数据修改 4.xml 数据删除 二. 1.xml 创建 import xml.etree.ElementT ...

  10. [剑指offer]10.斐波那契数列+青蛙跳台阶问题

    10- I. 斐波那契数列 方法一 Top-down 用递归实现 def fibonacci(n): if n <= 0: return 0 if n == 1: return 1 return ...