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; @Override public 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("语文"); //复制stu1 Student stu2 = (Student) stu1.clone(); System.out.println("学生1:" + stu1.number + " " + stu1.address + " " + stu1.courseList); System.out.println("学生2:" + stu2.number + " " + stu2.address + " " + stu2.courseList); //更改stu2 stu2.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; @Override public 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("语文"); //复制stu1 Student stu2 = (Student) stu1.clone(); System.out.println("学生1:" + stu1.number + " " + stu1.address + " " + stu1.courseList); System.out.println("学生2:" + stu2.number + " " + stu2.address + " " + stu2.courseList); //更改stu2 stu2.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") @Override public 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; } @Override public 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方法会触发对象里定义的 ...
随机推荐
- Python入门,新手之路
1.开始使用Python: 从上一篇中我们提到了第一个Python,print("Hello World!")程序说起,看到这一句话,你有没有想过如果括号中的语句不是Hello W ...
- 老oj曼哈顿最小生成树
Description 平面坐标系xOy内,给定n个顶点V = (x , y).对于顶点u.v,u与v之间的距离d定义为|xu – xv| + |yu – yv| 你的任务就是求出这n个顶点的最小生成 ...
- iOS9 未受信任的企业级开发者
升级iOS9,app打不开怎么办?6个步骤让你应对“未受信任的企业及开发者账号” 点开App,弹出未受信任的开发者,记住弹框中冒号后面的大写字母.关闭,进入设置. 进入通用 进入描述文件 找到所对应的 ...
- 酷炫地给py代码标上行数
Python IDLE是没有显示行号的功能的,今天学了一个方式可以酷炫地给自己的代码加上行号,该方法直接修改代码,慎用哦!代码如下: import fileinput for line in file ...
- Solr4.8.0源码分析(21)之SolrCloud的Recovery策略(二)
Solr4.8.0源码分析(21)之SolrCloud的Recovery策略(二) 题记: 前文<Solr4.8.0源码分析(20)之SolrCloud的Recovery策略(一)>中提 ...
- oracle 的一点累积
1. oracle用户相关 sqlplus sys/oracle as sysdba -- sys登录 create user xxx identified by password; -- ...
- HDOJ迷宫城堡(判断强连通 tarjan算法)
Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 65536/32768 K (Java/Others)Total Submission( ...
- POJ-3468-A Simple Problem with Integers(区间更新,求和)-splay或线段树
区间更新求和 主要用来练习splay树区间更新问题 //splay树的题解 // File Name: 3468-splay.cpp // Author: Zlbing // Created Time ...
- IOI 2009:Mecho
IOI2009 Mecho Time Limit: 10000ms Memory Limit: 262144KB This problem will be judged on SPOJ. Origin ...
- Selenium自动化测试环境搭建汇总(一):Selenium+Eclipse+Junit+TestNG
第一步 安装JDK JDk1.7. 下载地址:http://www.oracle.com/technetwork/java/javase/downloads/jdk7-downloads-188026 ...