我们自定义的类是以引用的形式放入集合,如果使用不当,会引发非常隐蔽的错误。就拿我经常问到的一个面试题来说明这个知识点。

第一步,我们定义一个Car类型的类,其中只有一个int类型id属性。

第二步,创建一个Car类的实例,假设是c,设置它的id是1。

第三步,我们通过new关键字创建两个不同的ArrayList,分别是a1和a2,这里请注意我的描述,是创建两个不同的ArrayList,而不是一个,并把第二步里创建的c对象分别放入a1和a2。

第四步,我们在a1这个ArrayList里,拿出c对象,并把它的id设置成2,同时不对存放在a2里的c对象做任何的修改。

我的问题是,完成上述四步后,a2里存放的c对象的id是多少?1还是2?

我们通过如下的CopyDemo.java来分析这个问题:

1    import java.util.ArrayList;
2 //第一步,创建一个Car类,其中只有一个属性i
3 //通过构造函数,我们可以设置i的值,而且有针对i的get和set方法
4 class Car {
1 private int i;
2 public int getI() {return i;}
3 public void setI(int i) {this.i = i; }
4 public Car(int i) {this.i = i; }
5 }
6 public class CopyDemo {
7 public static void main(String[] args) {
8 //第二步,创建一个Car的实例,其中的id是1
9 Car c = new Car(1);
10 //第三步,创建两个不同的ArrayList
11 ArrayList<Car> a1 = new ArrayList<Car>();
12 ArrayList<Car> a2 = new ArrayList<Car>();
13 //通过add方法把c分别加入到a1和a2里
14 a1.add(c);
15 a2.add(c);
16 //第四步,修改a1中的c对象,但不修改a2中的c对象
17 a1.get(0).setI(2);
18 //最后通过打印查看a2里c对象的id
19 System.out.println(a2.get(0).getI());
20 }
21 }

根据第19行的输出,虽然我们并没对a2里存放的c对象做任何的操作,但它的值也被改成了2,根据我面试下来的结果,估计有一半的初级程序员(工作经验3年以下)会回答错。

原因是我们之前提到过的:集合里存放的是引用。下面我们来详细说明这点。

第一,当执行完Car c = new Car(1);操作后,Java虚拟机会在内存里开辟一块空间存放id是1的c对象,假设这块空间的首地址是1000,那么c其实是指向1000号空间的引用。

第二,我们其实是把c这个引用放入到两个不同的ArrayList里,大家可以通过下图来观察下效果。

从上图里我们能看到,a1和a2里第一个元素存放的其实都是c这个引用。通过这个引用,都能指向到存放在1000号内存里id是1的这个c对象。

当我们通过a1修改存放在其中的c对象时,其实是通过c这个引用直接改变了1000号里的id,由于a2里存放的引用也是指向1000号内存,所以虽然我们并没有改变过a2里的c对象,但a2里的值也跟着变了。

在实际项目里,可能会遇到类似的问题。比如我们把同一份变量放入两个不同的集合对象里,我们的本意是,在一个集合里给该变量做个备份,只在另外一个集合里修改。但根据上文的描述,即使我们只在其中一个集合里做修改,这个修改会影响到另外一个我们企图做备份的集合,这和我们想当然的结果不一样。

如果要正确地实现“一个集合做备份另一个集合做修改”的效果,我们就得通过clone方法来实现深拷贝了,来看DeepCopy.java这个例子。

1    import java.util.ArrayList;
2 //实现Cloneable接口,重写其中的clone方法
3 class CarForDeepCopy implements Cloneable {
4 private int i;
5 public int getI() {return i;}
6 public void setI(int i) {this.i = i;}
7 //构造函数
8 public CarForDeepCopy(int i)
9 {this.i = i;}
10 //调用父类的clone完成对象的拷贝
11 public Object clone() throws CloneNotSupportedException
12 { return super.clone(); }
13 }
14 public class DeepCopy {
15 public static void main(String[] args) {
16 CarForDeepCopy c1 = new CarForDeepCopy(1);
17 //通过clone方法把c1做个备份
18 CarForDeepCopy c2 = null;
19 try {
20 c2 = (CarForDeepCopy)c1.clone();
21 } catch (CloneNotSupportedException e) {
22 e.printStackTrace();
23 }
24 ArrayList<CarForDeepCopy> a1 = new ArrayList<CarForDeepCopy>();
25 ArrayList<CarForDeepCopy> a2 = new ArrayList<CarForDeepCopy>();
26 a1.add(c1);
27 a2.add(c2);
28 //修改a1中的c对象的id为2
29 a1.get(0).setI(2);
30 //输出依然是1
31 System.out.println(a2.get(0).getI());
32 }
33 }

为了实现clone,我们自定义的类必须要像第3行那样实现Cloneable接口;同时像第11行那样重写clone方法。其中我们可以像第12行那样,通过super调用父类的clone方法来完成内容的拷贝。

在第20行里,我们通过调用c1对象的clone方法在内存里创建另外一个CarForDeepCopy对象,随后当我们通过第26和27行把c1和c2放入到两个ArrayList后,在内存里存储的结构如下图所示。

从中我们能看到,c1被clone后,系统会开辟一块新的空间用以存放和c1相同的内容,并用c2指向这块内存。

这样a1和a2两个ArrayList就通过c1和c2这两个不同的引用指向了两块不同的内存空间,所以a1的修改不会影响到a2对象。

集合中存的是引用,分析一道容易混淆的Java面试题的更多相关文章

  1. 【转】HashMap集合中key只能为引用数据类型,不能为基本类型

    在HashMap中,为什么不能使用基本数据类型作为key? 其实和HashMap底层的存储原理有关,HashMap存储数据的特点是:无序.无索引.不能存储重复元素. 存储元素采用的是hash表存储数据 ...

  2. 一道非常棘手的 Java 面试题:i++ 是线程安全的吗

    转载自  一道非常棘手的 Java 面试题:i++ 是线程安全的吗 i++ 是线程安全的吗? 相信很多中高级的 Java 面试者都遇到过这个问题,很多对这个不是很清楚的肯定是一脸蒙逼.内心肯定还在质疑 ...

  3. java集合中的HashMap源码分析

    1.hashMap中的成员分析 transient Node<K,V>[] table; //为hash桶的数量 /** * The number of key-value mapping ...

  4. 一道月薪3W的java面试题 (小明和小强都是张老师的学生,张老师的生日是某月某日,2人都不知道张老师的生日)

    小明和小强都是张老师的学生,张老师的生日是M月N日,2人都知道张老师的生日 是下列10组中的一天,张老师把M值告诉了小明,把N值告诉了小强,张老师问他们知道他的生日是那一天吗? 3月4日 3月5日 3 ...

  5. 一道简单到爆 Java面试题,居然挂了一票人

    很多时候bug往往都是出在,我们觉得非常简单,不起眼的基础知识上 年前公司最后一波招人,为年后项目做技术储备,主要招聘对象初中级Java开发,要求也并没有多苛刻,唯一一点基础稍好,快速上手做项目就行. ...

  6. Java EE中的容器和注入分析,历史与未来

    Java EE中的容器和注入分析,历史与未来 java中的容器 java中的注入 容器和注入的历史和展望 一.java中的容器 java EE中的注入,使我们定义的对象能够获取对资源和其他依赖项的引用 ...

  7. java代码实现将集合中的重复元素去掉

    package com.loaderman.test; import java.util.ArrayList; import java.util.LinkedHashSet; import java. ...

  8. Java去除ArrayList集合中重复字符串的案例

    ArrayList去除集合中的字符串重复值 分析: A:创建集合对象 B:添加多个字符串元素 C:创建新集合 D:遍历旧集合,获取得到每一个元素 E:拿着个元素到新集合去找,看有没有 有:不进去 没有 ...

  9. 一个关于集合的问题,为什么添加进List集合中的元素被莫名其妙的改变了

    以前自己理解的不够深刻,特此记录一下提醒自己,如果正好也帮到了你,我会很开心.相信只有自己正好遇到这个问题,才觉得哦,原来这样.自己小白,大神莫喷 为什么添加进List集合中的元素被莫名其妙的改变了? ...

随机推荐

  1. effective C++笔记-2

    6:析构函数使用 1.如果一个基类是为了多态用途,那么就应该有一个虚析构函数. 2.如果一个类中有虚函数,那么就应该就有一个虚的析构函数. 3.如果一个基类中不是为了多态的用途,或者不作为基类来使用, ...

  2. tcp/ip 卷一 读书笔记(3)为什么既要有IP地址又要有MAC地址

    网络层 首先明确一点,并不是所有的网络之间传输数据都需要mac地址和ip地址,比如说点对点线路之间的通信就没有MAC地址,网络层使用ipx协议时就没有ip地址,但是在当前的主流网络中,我们都使用ip地 ...

  3. SQL语句学习

    看似简单,但其实包含很多技巧思维 1.查询课程表中所有科目大于80的学生 select distinct name from student where name not in (select nam ...

  4. php 变量原理讲解

    php 变量原理讲解 一.变量概念   所谓变量,是指在程序中其值可以变化的量. 程序是管理和处理数据的.在程序运行过程中,我们需要存贮这些数据,变量和常量就是用于保存程序运行时的数据的. 变量通常由 ...

  5. HI3531编译helloworld,执行错误

    若在嵌入式系统中执行某文件出现如下错误: -/bin/sh: XXX: not found 一般是因为缺少库文件,解决方法有2: 1,文件系统的busybox编译时使用动态编译方式 2,或编译该文件的 ...

  6. zTree实现地市县三级级联DAO接口测试

    zTree实现地市县三级级联DAO接口测试 ProvinceDaoTest.java: /** * @Title:ProvinceDaoTest.java * @Package:com.gwtjs.d ...

  7. freemarker报错之五

    1.错误描述 freemarker.core.ParseException: Token manager error: freemarker.core.TokenMgrError: Lexical e ...

  8. angular路由参数说明

    AngularJS 路由 本章节我们将为大家介绍 AngularJS 路由. AngularJS 路由允许我们通过不同的 URL 访问不同的内容. 通过 AngularJS 可以实现多视图的单页Web ...

  9. HihoCoder - 1139

    在上一回和上上回里我们知道Nettle在玩<艦これ>,Nettle在整理好舰队之后终于准备出海捞船和敌军交战了.在这个游戏里面,海域是N个战略点(编号1..N)组成,如下图所示其中红色的点 ...

  10. iOS - MySQL 的安装配置

    前言 提前下载好相关软件,且安装目录最好安装在全英文路径下.如果路径有中文名,那么可能会出现一些莫名其妙的问题. 提前准备好的软件: mysql-5.7.17-macos10.12-x86_64.dm ...