博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
Java深、浅克隆(clone)
阅读量:6877 次
发布时间:2019-06-26

本文共 2151 字,大约阅读时间需要 7 分钟。

hot3.png

    Java的基类Object中有个本地方法clone(),由于该方法是protected访问权限,所以只能在Object内部或其子类内部访问,可是不能在类外通过对象.clone()访问。

protected native Object clone() throws CloneNotSupportedException;

    自定义的类需要实现"克隆",一定要实现Cloneable接口(该接口内没有任何方法,但如果不实现该接口就直接在类里调用Object的clone(),运行报错)。

    1、浅拷贝:

    需要实现拷贝的自定义User类

public class User implements Cloneable{		private int id;		private String name;		public User(int id, String name) {			this.id = id;			this.name = name;		}		@Override //这个注解有和没有都一样		public User clone() throws CloneNotSupportedException {			return (User)super.clone();//直接调用Object的clone(),并返回就可以了		}				public int getId() {			return id;		}				public String getName() {			return name;		}			}

    测试类:

public class TestClone {		public static void main(String[] args) throws CloneNotSupportedException, ClassNotFoundException, IOException {		User u1 = new User(1, "zhangsan");		User u2 = u1.clone();		System.out.println(u1==u2);		System.out.println(u1.getId()==u2.getId());		System.out.println(u1.getName()==u2.getName());	}}

    结果:false、true、true

    可能很多人会觉得第三个应该是false。我们先来看一下Object类提供的拷贝机制,就会明白了。

155228_IWoI_2708044.png

    Object类提供的Clone机制只对对象里的各实例变量进行“简单复制”,如果实例变量的类型是引用类型(本例中的String name),也只是简单得复制这个引用变量(应该是这样:u2=u1),这样u1和u2仍然指向内存中的同一实例。如果需要引用类型的实例也拷贝一份,那就需要深拷贝了。

    2、深拷贝:    

    需要拷贝的User类还要实现 Serializable接口(也是没有任何方法的接口),因为要用到输入、输出流将对象实例从内存中复制一份。

    User类:

public class User implements Cloneable,Serializable{		private int id;		private String name;				@Override		public User clone() throws CloneNotSupportedException {		    ByteArrayOutputStream baos = new ByteArrayOutputStream();//字节输出流			ObjectOutputStream out = new ObjectOutputStream(baos);//对象输出流中嵌套封装字节输出流						out.writeObject(this);//将调用clone()当前对象的数据输出保存在字节输出流中						ByteArrayInputStream bais = new ByteArrayInputStream(baos.toByteArray());//字节输入流,指定读取的数据来自baos中的字节			ObjectInputStream in = new ObjectInputStream(bais);//对象输入流封装字节输入流						User u = (User)in.readObject();//将数据以对象的形式读取			return u;		}				public User(int id, String name) {			super();			this.id = id;			this.name = name;		}				public int getId() {			return id;		}		public String getName() {			return name;		}	}

    测试类还是和上面的测试类一样,结果为:false,true,false

通过下图便可了解深拷贝的机制:

161705_f0IT_2708044.png

 

转载于:https://my.oschina.net/henryking/blog/781826

你可能感兴趣的文章
sizeof & strlen 字符串问题
查看>>
Tag文件和Tag标记
查看>>
Android studio 导入以前eclipse项目 编码设置
查看>>
Yii2 Day 10 创建命令行应用
查看>>
java容器源码分析(八)——LinkedHashSet
查看>>
linux中的内存
查看>>
Linux查看磁盘挂载信息和卸载方式
查看>>
maven 乱码处理
查看>>
mapreduce文件读取与清洗
查看>>
topcoder Single Round 624(500)
查看>>
字符串匹配的Boyer-Moore算法
查看>>
Memcache 简单操作
查看>>
获取微信用户信息
查看>>
CentOS 7修改网络接口名称
查看>>
Javascript博客转载集合
查看>>
ActiveMQ学习笔记06 - 消费者负载均衡与高可用
查看>>
servlet线程不安全的实例已及解决办法
查看>>
logstash使用
查看>>
Oracle从表的中文注释创建中文视图
查看>>
eclipse_集成Properties_Editor
查看>>