Java中的浅复制和深复制 Cloneable clone
先看一个简单案例
public class Test {public static void main(String args[]) {Student stu1 = new Student();stu1.number = 1;Student stu2 = stu1;//stu1和stu2指向堆内存中同一个对象System.out.println("学生1:" + stu1.number + " 学生2:" + stu2.number);stu2.number = 2;System.out.println("学生1:" + stu1.number + " 学生2:" + stu2.number);System.out.println(stu1 == stu2);}}class Student {public int number;}如上这种形式,仅仅将一个对象的引用赋给另一个引用,并没有复制出另一个对象,这两个引用指向的是内存中的同一个对象所以,这根本不叫对象的复制
对象的复制(浅复制)
public class Test {public static void main(String args[]) {Student stu1 = new Student();stu1.number = 1;Student stu2 = (Student) stu1.clone();System.out.println("学生1:" + stu1.number + " 学生2:" + stu2.number);stu2.number = 2;System.out.println("学生1:" + stu1.number + " 学生2:" + stu2.number);System.out.println(stu1 == stu2);}}class Student implements Cloneable {public int number;@Overridepublic Object clone() {Student stu = null;try {stu = (Student) super.clone();} catch (CloneNotSupportedException e) {e.printStackTrace();}return stu;}}Object有个protected的clone方法,该方法的签名是:protected native Object clone() throws CloneNotSupportedException;要想对一个对象进行复制,就需要覆盖clone方法,一般步骤是(浅复制):
- 1、被复制的类需要实现Clonenable接口,不实现的话在调用clone方法会抛出CloneNotSupportedException异常,该接口为标记接口(不含任何方法)
- 2、覆盖clone()方法,访问修饰符设为public。方法中调用super.clone()方法得到需要的复制对象(native为本地方法)
浅复制可能出现的问题
public class Test {public static void main(String args[]) {Student stu1 = new Student();stu1.number = 1;stu1.address = new Address("广州");stu1.courseList.add("语文");//复制stu1Student stu2 = (Student) stu1.clone();System.out.println("学生1:" + stu1.number + " " + stu1.address + " " + stu1.courseList);System.out.println("学生2:" + stu2.number + " " + stu2.address + " " + stu2.courseList);//更改stu2stu2.number = 2;stu2.address = new Address("深圳");stu2.courseList.add("数学");System.out.println("学生1:" + stu1.number + " " + stu1.address + " " + stu1.courseList);System.out.println("学生2:" + stu2.number + " " + stu2.address + " " + stu2.courseList);}}class Student implements Cloneable {public int number;public ArrayList<String> courseList = new ArrayList<String>();public Address address;@Overridepublic Object clone() {Student stu = null;try {stu = (Student) super.clone();} catch (CloneNotSupportedException e) {e.printStackTrace();}return stu;}}class Address {public String add;public Address(String add) {this.add = add;}}更改stu2的address和courseList后,是不是发现,stu1的相应值也变了?这是为什么呢?这是因为Java做了一个偷懒的拷贝动作,Object类提供的方法clone只是拷贝本对象,其对象内部的数组、引用对象等都不拷贝,还是指向原生对象的内部元素的地址,这种拷贝就叫做浅拷贝。确实是非常浅,两个对象共享了一个私有变量,你改我改大家都能改,是一个种非常不安全的方式,在实际项目中使用还是比较少的。注意:八大基本类型和String(虽然String 是引用类型,但Java其实是希望你把它也"认为"是基本类型,String 是没有 clone 方法的)等都会被拷贝的。
深复制
public class Test {public static void main(String args[]) {Student stu1 = new Student();stu1.number = 1;stu1.address = new Address("广州");stu1.courseList.add("语文");//复制stu1Student stu2 = (Student) stu1.clone();System.out.println("学生1:" + stu1.number + " " + stu1.address + " " + stu1.courseList);System.out.println("学生2:" + stu2.number + " " + stu2.address + " " + stu2.courseList);//更改stu2stu2.number = 2;stu2.address = new Address("深圳");stu2.courseList.add("数学");System.out.println("学生1:" + stu1.number + " " + stu1.address + " " + stu1.courseList);System.out.println("学生2:" + stu2.number + " " + stu2.address + " " + stu2.courseList);}}class Student implements Cloneable {public int number;public ArrayList<String> courseList = new ArrayList<String>();public Address address;@SuppressWarnings("unchecked")@Overridepublic Object clone() {Student stu = null;try {stu = (Student) super.clone();//浅复制stu.address = (Address) address.clone(); //深度复制stu.courseList = (ArrayList<String>) courseList.clone();//深度复制} catch (CloneNotSupportedException e) {e.printStackTrace();}return stu;}}class Address implements Cloneable {//为了达到真正的复制对象,而不是纯粹引用复制。我们需要将Address类可复制化public String add;public Address(String add) {this.add = add;}@Overridepublic Object clone() {Address addr = null;try {addr = (Address) super.clone();} catch (CloneNotSupportedException e) {e.printStackTrace();}return addr;}}
Java中的浅复制和深复制 Cloneable clone的更多相关文章
- JAVA中浅复制与深复制 - coolmist - ITeye技术网站
body{ font-family: "Microsoft YaHei UI","Microsoft YaHei",SimSun,"Segoe UI& ...
- Java中引用的浅复制和深复制
Java中除了基本类型int,char,double等的赋值是按照值传递之外,其余的类型和对象都是按照引用进行传递的. 下面来看一个关于引用的例子. package referenceCopy;// ...
- JAVA中浅复制与深复制
1.浅复制与深复制概念⑴浅复制(浅克隆)被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象.换言之,浅复制仅仅复制所考虑的对象,而不复制它所引用的对象. ⑵深复 ...
- Java中的“浅复制”与“深复制”
复制 将一个对象的引用复制给另一个对象,一共有三种方式.第一种方式是直接赋值,第二种方式是浅复制,第三种方式是深复制. 1.直接赋值 在Java中,A a1 = a2,这实际上复制的是引用,也就是说 ...
- java基础-浅复制与深复制的理解
浅复制与深复制在很多编程语言中都有出现,那么什么是浅复制,什么是深复制呢? 要区分浅复制与深复制,首先我们要明确什么是复制,怎样才算是复制.复制的例子在生活中也随处可见,如复印一份文档,复制一段文字等 ...
- Java里的浅复制与深复制
1.浅复制与深复制概念 ⑴浅复制(浅克隆) 被复制对象的所有变量都含有与原来的对象相同的值,而所有的对其他对象的引用仍然指向原来的对象.换言之,浅复制仅仅复制所考虑的对象,而不 复制它所引用的对象. ...
- 深度解析javascript中的浅复制和深复制
原文:深度解析javascript中的浅复制和深复制 在谈javascript的浅复制和深复制之前,我们有必要在来讨论下js的数据类型.我们都知道有Number,Boolean,String,Null ...
- js中的浅复制和深复制
浅复制:浅复制是复制引用,复制后的引用都是指向同一个对象的实例,彼此之间的操作会互相影响 深复制:深复制不是简单的复制引用,而是在堆中重新分配内存,并且把源对象实例的所有属性都进行新建复制,以保证深复 ...
- Python中的浅复制、深复制
参考 https://docs.python.org/3/library/copy.html?highlight=copy%20copy#copy.copy https://en.wikipedia. ...
- php对象复制、clone、浅复制与深复制实例详解
php对象复制.clone.浅复制与深复制实例详解 一.用clone(克隆)来复制对象$obj1 = new Object();$obj2 = clone $obj1;clone方法会触发对象里定义的 ...
随机推荐
- C# 窗体间传值方法大汇总(转)
第一种方法:创建一个类,里面声明用于存储接收的字段.传的时候存储于字段中,要用的时候,直接类名.字段名 进行调用.(这种方法传递是双向的) 第二种方法:1.在Form1里定义 public strin ...
- yum版本新增包的一般步骤
在Jekins的自动构建环境中,有时会有在构建出的ISO中添加新应用app需求,对于采用rpm包源代码管理方式的构建环境来说,基本步骤如下: 1.下载app的src.rpm包 2.解压src.rpm包 ...
- ASP.NET MVC 定义JsonpResult实现跨域请求
1:原理 在js中,XMLHttpRequest是不能请求不同域的数据,但是script标签却可以,所以可以用script标签实现跨域请求.具体是定义一个函数,例如jsonp1234,请求不同域的ur ...
- PLSQL Developer如何设置自动打开上次编辑的文件
作为开发人员经常把sql语句保存到文件中以方便下次继续使用,问题是plsqlDev重启后每次都需要手工打开这个文件,好不方便: 以下设置是plsqlDev启动后自动打开上次编辑的文件. 选择配置> ...
- Python使用纯真年代数据库qqwry.dat转换物理位置
PS:网上直接找的,贴出来,方便以后随时用,感谢分享的人. #!/usr/bin/python #encoding: utf-8 import socket import codecs import ...
- c++ 顺序容器学习 - 容器适配器
摘要: 对 容器适配器 的疑问. 刚开始接触 容器适配器 时,总感觉怪怪的,认为多此一举,顺手搜了搜,原来我在这一点is not alone: STL容器适配器的用途 其中有个老兄说的好,这里 引用一 ...
- 从linux内核中学到的编程技巧 【转】
从linux内核中学到的编程技巧 分类: LINUX 1构建泛型宏 (./linux/include/linux/kernel.h) #define min(x, y) ({ \ typeof(x ...
- RH6030 单通道触摸感应开关
1.概述: RH6030 是一款单通道电容式触摸感应控制开关IC,可以替代传统的机械式开关. 该 IC 采用CMOS 工艺制造,结构简单,性能稳定.IC 可通过外部引脚配置成多种工作模式,可广泛应用于 ...
- PlatformTransactionManager
Spring Boot 使用事务非常简单,首先使用注解 @EnableTransactionManagement 开启事务支持后,然后在访问数据库的Service方法上添加注解 @Transactio ...
- Qt入门(16)——组装窗口部件
这个例子显示了创建几个窗口部件并用信号和槽把它们连接起来,和如何处理重新定义大小事件. #include <qapplication.h> #include <qpushbutton ...



