谈到了对象的克隆,就不得不说为什么要对对象进行克隆。Java中所有的对象都是保存在堆中,而堆是供全局共享的。也就是说,如果同一个Java程序的不同方法,只要能拿到某个对象的引用,引用者就可以随意的修改对象的内部数据(前提是这个对象的内部数据通过get/set方法曝露出来)。有的时候,我们编写的代码想让调用者只获得该对象的一个拷贝(也就是一个内容完全相同的对象,但是在内存中存在两个这样的对象),有什么办法可以做到呢?当然是克隆咯。
import java.io.Serializable; public class Apple implements Cloneable,Serializable { private static final long serialVersionUID = 1L; private String locality; public String getLocality() { return locality; } public void setLocality(String locality) { this.locality = locality; } @Override protected Object clone() throws CloneNotSupportedException { return super.clone(); } }
上面的代码,简单的构建了一个Apple对象,并实现了Cloneable及Serializable接口,并重写clone()方法。
接着再构建一个Fruit对象,里面有一个Apple属性。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 | import java.io.Serializable; public class Fruit implements Cloneable,Serializable{ private static final long serialVersionUID = 1L; private String fruitName; private Apple apple; public Apple getApple() { return apple; } public void setApple(Apple apple) { this .apple = apple; } public String getFruitName() { return fruitName; } public void setFruitName(String fruitName) { this .fruitName = fruitName; } @Override protected Fruit clone() throws CloneNotSupportedException { return (Fruit)super.clone(); } } |
接着复制一下对象:
1 2 3 4 5 6 7 | Fruit fruit = new Fruit(); fruit.setFruitName( "Apple" );fruit.setApple(apple); Fruit fruit2 = fruit.clone(); System. out .println( "Fruit = " +(fruit==fruit2)); System. out .println( "Fruit's apple=" +(fruit.getApple()==fruit2.getApple()));<br><br> |
Apple apple = new Apple();
apple.setLocality("China");
Fruit = false
Fruit's apple=true
很明显,这里的复制是浅复制(shallow clone)。fruit2内的apple属性与fruit内的apple属性是同一个对象,只复制了引用的地址。
而深克隆,就是非浅克隆。克隆除自身以外所有的对象,包括自身所包含的所有对象实例。至于深克隆的层次,由具体的需求决定,也有“N层克隆”一说。
但是,所有的基本(primitive)类型数据,无论是浅克隆还是深克隆,都会进行原值克隆。毕竟他们都不是对象,不是存储在堆中。注意:基本数据类型并不包括他们对应的包装类。
如果我们想让对象进行深度克隆,我们可以这样修改Fruit类的clone()方法。
1 2 3 4 5 6 7 | @Override protected Fruit clone() throws CloneNotSupportedException { Fruit fruit = (Fruit)super.clone(); fruit.apple = (Apple)fruit.getApple().clone(); return fruit; } |
这样就可以实现了,如果类层次较深,这样写就很麻烦。
接下来要重点介绍一下使用java.lang.Serializable来实现对象的深度克隆。
首先,我们编写一个工具类并提供cloneTo()方法。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 | import java.io.ByteArrayInputStream; import java.io.ByteArrayOutputStream; import java.io.IOException; import java.io.ObjectInputStream; import java.io.ObjectOutputStream; public class BeanDeepCloneUtil { @SuppressWarnings( "unchecked" ) public static <T> T cloneTo(T src) { ByteArrayOutputStream memoryBuffer = new ByteArrayOutputStream(); ObjectOutputStream outputStream = null ; ObjectInputStream inputStream = null ; T destination = null ; try { outputStream = new ObjectOutputStream(memoryBuffer); outputStream.writeObject(src); outputStream.flush(); inputStream = new ObjectInputStream( new ByteArrayInputStream( memoryBuffer.toByteArray())); destination = (T) inputStream.readObject(); } catch (IOException e) { // TODO Auto-generated catch block e.printStackTrace(); } catch (ClassNotFoundException e) { // TODO Auto-generated catch block e.printStackTrace(); } finally { try { if (outputStream != null ) { outputStream.close(); outputStream= null ; } } catch (IOException e) { e.printStackTrace(); } try { if (inputStream != null ) { inputStream.close(); inputStream= null ; } } catch (IOException e) { e.printStackTrace(); } } return destination; } } |
接下来我们测试一下是否能通过这个工具来实现深度克隆。
1 2 3 4 5 6 7 8 9 10 11 | Apple apple = new Apple(); apple.setLocality( "China" ); Fruit fruit = new Fruit(); fruit.setFruitName( "Apple" ); fruit.setApple(apple); Fruit fruit3 = BeanDeepCloneUtil.cloneTo(fruit); System. out .println( "Fruit = " +(fruit==fruit3)); System. out .println( "Fruit's apple=" +(fruit.getApple()==fruit3.getApple())); |
运行结果如下:
Fruit = false
Fruit's apple=false
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 25岁的心里话
· 闲置电脑爆改个人服务器(超详细) #公网映射 #Vmware虚拟网络编辑器
· 零经验选手,Compose 一天开发一款小游戏!
· 因为Apifox不支持离线,我果断选择了Apipost!
· 通过 API 将Deepseek响应流式内容输出到前端