`
ttkktt
  • 浏览: 27313 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

深入谈谈java的深拷贝与浅拷贝

阅读更多

首先我们看看浅拷贝和深拷贝的定义

浅拷贝:只复制一个对象,对象内部存在的指向其他对象数组或者引用则不复制

深拷贝:对象,对象内部的引用均复制

为了更好的理解它们的区别我们假设有一个对象A,它包含有2对象对象A1和对象A2(图1)


 

对象A进行浅拷贝后,得到对象B但是对象A1和A2并没有被拷贝(图2)


 

对象A进行深拷贝,得到对象B的同时A1和A2连同它们的引用也被拷贝(图3)


 

在理解了深拷贝和浅拷贝后,我们来看看java的深拷贝和浅拷贝实现。java.lang.Object的clone()方法默认是返回一个前拷贝对象。因此如果要用clone()方法实现一个深拷贝,我们必须对每个对象的clone()方法进行特别实现。当对象层次复杂的时候,这样做不但困难而且浪费时间和容易出现错误,特别有时候你不但需要深拷贝同时你也对这个对象进行浅拷贝的时候,你会发现写这个clone()方法真不是一个好的解决方案。

 

那么除了clone()方法,我们还可以怎么实现呢?答案是序列化,实现步骤和思路是把要拷贝的对象输出成byte array,然后再利用ObjectInputStream转换出新的对象。下面是代码

	public static Object copy(Object oldObj) {
		Object obj = null;
		try {
			// Write the object out to a byte array
			ByteArrayOutputStream bos = new ByteArrayOutputStream();
			ObjectOutputStream out = new ObjectOutputStream(bos);
			out.writeObject(oldObj);
			out.flush();
			out.close();

			// Retrieve an input stream from the byte array and read
			// a copy of the object back in.
			ByteArrayInputStream bis = new ByteArrayInputStream(bos.toByteArray()); 
			ObjectInputStream in = new ObjectInputStream(bis);
			obj = in.readObject();
		} catch (IOException e) {
			e.printStackTrace();
		} catch (ClassNotFoundException cnfe) {
			cnfe.printStackTrace();
		}
		return obj;
	}
 

 

  • 大小: 5.1 KB
  • 大小: 7 KB
  • 大小: 9.8 KB
2
0
分享到:
评论
3 楼 e421083458 2013-04-22  
不错!  
2 楼 ttkktt 2009-05-21  
to
TonyLian 写道

博主(也是绝大部分人)所说的“浅拷贝”,我认为叫“影子拷贝”更为贴切。代之的【浅拷贝】应该是指原生类型(可以也包括非可变型,如String),进行拷贝,而数组或引用,不做任何处理。与“影子拷贝”的区别即在此,不做任何处理,==赋值 都不做(而【影子拷贝】是要做的),目的是保护性的,避免“误操作”。如博主的图中,A和B中域都指向A1和A2,这是一件很危险的事情。

另外,博主给出的这一段代码,应该是通过流来“克隆”,而非序列化/反序列化。相比之,序列化的应用面更为狭窄,因为被“克隆”的类必须是要实现序列化接口的。

可以看看《Effective Java》或者http://tonylian.iteye.com/admin/blogs/383541




你没看完整我的文章吧,深拷贝的图是图3。。。,另外序列化就是把对象变成字节流。
之所以用序列化来做深拷贝而不是用clone方法来做深拷贝我文章也说明了,是因为当你的对象层次很复杂的时候,你对每个对象写这个clone方法耗费的时间和困难度是很高的而且容易出错。
1 楼 TonyLian 2009-05-20  
博主(也是绝大部分人)所说的“浅拷贝”,我认为叫“影子拷贝”更为贴切。代之的【浅拷贝】应该是指原生类型(可以也包括非可变型,如String),进行拷贝,而数组或引用,不做任何处理。与“影子拷贝”的区别即在此,不做任何处理,==赋值 都不做(而【影子拷贝】是要做的),目的是保护性的,避免“误操作”。如博主的图中,A和B中域都指向A1和A2,这是一件很危险的事情。

另外,博主给出的这一段代码,应该是通过来“克隆”,而非序列化/反序列化。相比之,序列化的应用面更为狭窄,因为被“克隆”的类必须是要实现序列化接口的。

可以看看《Effective Java》或者http://tonylian.iteye.com/admin/blogs/383541

相关推荐

Global site tag (gtag.js) - Google Analytics