本文是“最最最常见Java面试题总结”系列第一篇文章。 主要内容是:值传递和引用传递、==与equals、 hashCode与equals。
首先要明确的是:“对象传递(数组、类、接口)是引用传递,原始类型数据(整型、浮点型、字符型、布尔型)传递是值传递。”
值传递是指对象被值传递,意味着传递了对象的一个副本,即使副本被改变,也不会影响源对象。(因为值传递的时候,实际上是将实参的值复制一份给形参。)
引用传递是指对象被引用传递,意味着传递的并不是实际的对象,而是对象的引用。因此,外部对引用对象的改变会反映到实际的对象上。(因为引用传递的时候,实际上是将实参的地址值复制一份给形参。)
有时候面试官不是单纯问你“Java中是值传递还是引用传递”是什么啊,而是给出一个例子,然后让你写出答案,这种也常见在笔试题目中!所以,非常重要了,请看下面的例子:
public static void main(String[] args) { int num1 = 10; int num2 = 20; swap(num1, num2); System.out.println("num1 = " + num1); System.out.println("num2 = " + num2); } public static void swap(int a, int b) { int temp = a; a = b; b = temp; System.out.println("a = " + a); System.out.println("b = " + b); }
结果:
a = 20 b = 10 num1 = 10 num2 = 20
解析:
在swap方法中,a、b的值进行交换,并不会影响到num1、num2。因为,a、b中的值,只是从num1、num2的复制过来的。 也就是说,a、b相当于num1、num2的副本,副本的内容无论怎么修改,都不会影响到原件本身。
public static void main(String[] args) { int[] arr = {1,2,3,4,5}; change(arr); System.out.println(arr[0]); } public static void change(int[] array) { //将数组的第一个元素变为0 array[0] = 0; }
结果:
1 0
解析:
无论是主函数,还是change方法,操作的都是同一个地址值对应的数组。 因此,外部对引用对象的改变会反映到实际的对象上。
// 测试引用传递:StringBuffer @org.junit.Test public void method1() { StringBuffer str = new StringBuffer("公众号:Java面试通关手册"); System.out.println(str); change1(str); System.out.println(str); } public static void change1(StringBuffer str) { str = new StringBuffer("abc");//输出:“公众号:Java面试通关手册” //str.append("欢迎大家关注");//输出:公众号:Java面试通关手册欢迎大家关注 //str.insert(3, "(编程)");//输出:公众号(编程):Java面试通关手册 }
结果:
公众号:Java面试通关手册 公众号:Java面试通关手册
解析:
很多要这个时候要问了:StringBuffer创建的明明也是对象,那为什么输出结果依然是原来的值呢?
因为在 change1
方法内部我们是新建了一个StringBuffer对象,所以 str
指向了另外一个地址,相应的操作也同样是指向另外的地址的。
那么,如果将 change1
方法改成如下图所示,想必大家应该知道输出什么了,如果你还不知道,那可能就是我讲的有问题了,我反思(开个玩笑,实际上,上面程序中已经给出答案):
public static void change1(StringBuffer str) { str.append("欢迎大家关注"); str.insert(3, "(编程)"); }
// 测试引用传递:Sring @org.junit.Test public void method2() { String str = new String("公众号:Java面试通关手册"); System.out.println(str); change2(str); System.out.println(str); } public static void change2(String str) { // str="abc"; //输出:公众号:Java面试通关手册 str = new String("abc"); //输出:公众号:Java面试通关手册 }
结果:
公众号:Java面试通关手册 公众号:Java面试通关手册
可以看到不论是执行 str="abc;"
还是 str=newString("abc");
str的输出的值都不变。 按照我们上面讲“StringBuffer类型传递”的时候说的, str="abc;"
应该会让str的输出的值都不变。为什么呢?因为String在创建之后是不可变的。
下面的程序输出是什么?
public class Demo { public static void main(String[] args) { Person p = new Person("张三"); change(p); System.out.println(p.name); } public static void change(Person p) { Person person = new Person("李四"); p = person; } } class Person { String name; public Person(String name) { this.name = name; } }
很明显仍然会输出 张三
。因为 change
方法中重新创建了一个 Person
对象。
那么,如果把 change
方法改为下图所示,输出结果又是什么呢?
public static void change(Person p) { p.name="李四"; }
答案我就不说了,我觉得大家如果认真看完上面的内容之后应该很很清楚了。
== : 它的作用是判断两个对象的地址是不是相等。即,判断两个对象是不是同一个对象。(基本数据类型==比较的是值,引用数据类型==比较的是内存地址)
equals() : 它的作用也是判断两个对象是否相等。但它一般有两种使用情况:
举个例子:
public class test1 { public static void main(String[] args) { String a = new String("ab"); // a 为一个引用 String b = new String("ab"); // b为另一个引用,对象的内容一样 String aa = "ab"; // 放在常量池中 String bb = "ab"; // 从常量池中查找 if (aa == bb) // true System.out.println("aa==bb"); if (a == b) // false,非同一对象 System.out.println("a==b"); if (a.equals(b)) // true System.out.println("aEQb"); if (42 == 42.0) { // true System.out.println("true"); } } }
上述运行结果说明:
面试官可能会问你:“你重写过 hashcode 和 equals 么,为什么重写equals时必须重写hashCode方法?”
hashCode() 的作用是获取哈希码,也称为散列码;它实际上是返回一个int整数。这个哈希码的作用是确定该对象在哈希表中的索引位置。hashCode() 定义在JDK的Object.java中,这就意味着Java中的任何类都包含有hashCode() 函数。
散列表存储的是键值对(key-value),它的特点是:能根据“键”快速的检索出对应的“值”。这其中就利用到了散列码!(可以快速找到所需要的对象)
我们以“HashSet如何检查重复”为例子来说明为什么要有hashCode:
当你把对象加入HashSet时,HashSet会先计算对象的hashcode值来判断对象加入的位置,同时也会与其他已经加入的对象的hashcode值作比较,如果没有相符的hashcode,HashSet会假设对象没有重复出现。但是如果发现有相同hashcode值的对象,这时会调用equals()方法来检查hashcode相等的对象是否真的相同。如果两者相同,HashSet就不会让其加入操作成功。如果不同的话,就会重新散列到其他位置。(摘自我的Java启蒙书《Head fist java》第二版)。这样我们就大大减少了equals的次数,相应就大大提高了执行速度。
如果觉得文章不错,欢迎转发点赞。你的转发与点赞就是我原创最大的动力!
本文分享自微信公众号 - 一个程序员的成长(xiaozaibuluo),作者:Snailclimb
原文出处及转载信息见文内详细说明,如有侵权,请联系 yunjia_community@tencent.com 删除。
原始发表时间:2018-08-24
本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。
我来说两句