收集大量Java经典面试题目📚,内容涵盖了包括了:Java面试、Spring、JVM、MyBatis、Redis、MySQL、并发编程、微服务、Linux、Springboot、SpringCloud、MQ、Kafka 等知识点🏝️。适合准备Java面试的读者参考和复习🌟📢。
❗ ❗ ❗
关注公众号:枫蜜柚子茶 ✅✅
🗳
📑 回 复 “Java面试” 获 取 完 整 资 料⬇ ⬇ ⬇
📖Java经典基础面试题目Top90道题🔥🔥
1️⃣ Java 概 述
2️⃣ 基 础 语 法
3️⃣ 面 向 对 象 🚩
4️⃣ IO 流
5️⃣ 反 射
6️⃣ API 使 用
一、对象相等判断
1. == 和 equals 的区别是什么
- ◾ == : 它的作用是判断两个对象的地址是不是相等。即,判断两个对象是不是同一个对象。 (基本数 据类型 == 比较的是值,引用数据类型 == 比较的是内存地址)
- ◾ equals() : 它的作用也是判断两个对象是否相等。但它一般有两种使用情况:
- ▪ 情况1:类没有覆盖 equals() 方法。则通过 equals() 比较该类的两个对象时,等价于通过 “==”比较这两个对象。
- ▪ 情况2:类覆盖了 equals() 方法。一般,我们都覆盖 equals() 方法来两个对象的内容相等; 若它们的内容相等,则返回 true (即,认为这两个对象相等)。
- ▪ 情况1:类没有覆盖 equals() 方法。则通过 equals() 比较该类的两个对象时,等价于通过 “==”比较这两个对象。
- ◾ 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"); }
} }
说明:
🔹 String中的equals方法是被重写过的,因为object的equals方法是比较的对象的内存地址, 而String的equals方法比较的是对象的值。
🔹 当创建String类型的对象时,虚拟机会在常量池中查找有没有已经存在的值和要创建的值相 同的对象,如果有就把它赋给当前引用。如果没有就在常量池中重新创建一个String对象。
2. 对象的相等与指向他们的引用相等,两者有什么不同?
对象的相等 比的是内存中存放的内容是否相等而 引用相等 比较的是他们指向的内存地址是否相 等。
3. hashCode 与 equals (重要)
HashSet如何检查重复 ❓
- ▪ 两个对象的 hashCode() 相同,则 equals() 也一定为 true,对吗? . hashCode和equals方法的关系
- ▪ 面试官可能会问你: “你重写过 hashcode 和 equals 么,为什么重写equals时必须重写hashCode 方法?ℽ
hashCode()介绍
- ⭕ hashCode() 的作用是获取哈希码,也称为散列码;它实际上是返回一个int整数。这个哈希码的作 用是确定该对象在哈希表中的索引位置。 hashCode() 定义在JDK的Object.java中,这就意味着Java中的任何类都包含有hashCode()函数。
- ⭕ 散列表存储的是键值对(key-value),它的特点是:能根据“键”快速的检索出对应的“值”。这其中就 利用到了散列码!(可以快速找到所需要的对象)
为什么要有 hashCode
我们以“HashSet 如何检查重复”为例子来说明为什么要有 hashCode:
🔸 当你把对象加入 HashSet 时, HashSet 会先计算对象的 hashcode 值来判断对象加入的位置,同 时也会与其他已经加入的对象的 hashcode 值作比较,如果没有相符的hashcode , HashSet会假 设对象没有重复出现。但是如果发现有相同 hashcode 值的对象,这时会调用 equals()方法来检查 hashcode 相等的对象是否真的相同。如果两者相同, HashSet 就不会让其加入操作成功。如果不 同的话,就会重新散列到其他位置。(摘自我的Java启蒙书《Head first java》第二版)。这样我 们就大大减少了 equals 的次数,相应就大大提高了执行速度。
hashCode()与equals()的相关规定
- ◾ 如果两个对象相等,则hashcode一定也是相同的
- ◾ 两个对象相等,对两个对象分别调用equals方法都返回true . 两个对象有相同的hashcode值,它们也不一定是相等的。因此,equals 方法被覆盖过,则 hashCode 方法也必须被覆盖hashCode() 的默认行为是对堆上的对象产生独特值。如果没有重写 hashCode(),则该 class 的两个对象无论如何都不会相等(即使这两个对象指向相同的数据)。
二、值传递
4. 当一个对象被当作参数传递到一个方法后,此方法可改变这个对象的属性,并 可返回变化后的结果,那么这里到底是值传递还是引用传递 ❓❗
是值传递。Java 语言的方法调用只支持参数的值传递。当一个对象实例作为一个参数被传递到方法中时,参数的值就是对该对象的引用。对象的属性可以在被调用过程中被改变,但对对象引用的改变是不会影响到调用者的。
5. 为什么 Java 中只有值传递
首先回顾一下在程序设计语言中有关将参数传递给方法(或函数)的一些专业术语。 按值调用
(call by value)表示方法接收的是调用者提供的值,而按引用调用(call by reference)表示方法接收的是调用者提供的变量地址。 一个方法可以修改传递引用所对应的变量值,而不能修改传递值调用所对应的变量值。 它用来描述各种程序设计语言(不只是Java)中方法参数传递方式。
Java程序设计语言总是采用按值调用。也就是说,方法得到的是所有参数值的一个拷贝,也就是说,方法不能修改传递给它的任何参数变量的内容。
🔻 下面通过 3 个例子来给大家说明:
example 1
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 的副本,副本的内容无论怎 么修改,都不会影响到原件本身。
通过上面例子,我们已经知道了一个方法不能修改一个基本数据类型的参数,而对象引用作为参数就不一样,请看
example 2
public static void main(String[] args) { int[] arr = { 1, 2, 3, 4, 5 };
System.out.println(arr[0]); change(arr);
System.out.println(arr[0]); }
public static void change(int[] array) { // 将数组的第一个元素变为0
array[0] = 0; }
结果:
1 0
解析:
array 被初始化 arr 的拷贝也就是一个对象的引用,也就是说 array 和 arr 指向的时同一个数组对 象。 因此,外部对引用对象的改变会反映到所对应的对象上。
通过 example2 我们已经看到,实现一个改变对象参数状态的方法并不是一件难事。理由很简单,方法得到的是对象引用的拷贝,对象引用及其他的拷贝同时引用同一个对象。
很多程序设计语言(特别是,C++和Pascal)提供了两种参数传递的方式:值调用和引用调用。有些程序员(甚至一定的普遍性,所以下面给出一个反例来详细地阐述一下这个问题。
example 3
public class Test {
public static void main(String[] args) { // TODO Auto-generated method stub
Student s1 = new Student("小张");
Student s2 = new Student("小李"); Test.swap(s1, s2);
System.out.println("s1:" + s1.getName()); System.out.println("s2:" + s2.getName());
}
public static void swap(Student x, Student y) { Student temp = x;
x = y;
y = temp;
System.out.println("x:" + x.getName()); System.out.println("y:" + y.getName());
} }
结果:
x:小李 y:小张 s1:小张 s2:小李
解析:
交换之前:
交换之后:
通过上面两张图可以很清晰的看出:
方法并没有改变存储在变量 s1 和 s2 中的对象引用。swap方法
的参数x和y被初始化为两个对象引用的拷贝,这个方法交换的是这两个拷贝
总结💡
Java程序设计语言对对象采用的不是引用调用,实际上,对象引用是按值传递的。
下面再总结一下Java中方法参数的使用情况:
✅ 一个方法不能修改一个基本数据类型的参数(即数值型或布尔型》 。 一个方法可以改变一个对象参数的状态。
✅ 一个方法不能让对象参数引用一个新的对象。
6. 值传递和引用传递有什么区别
1️⃣值传递:指的是在方法调用时,传递的参数是按值的拷贝传递,传递的是值的拷贝,也就是说传递 后就互不相关了。
2️⃣引用传递:指的是在方法调用时,传递的参数是按引用进行传递,其实传递的引用的地址,也就是 变量所对应的内存空间的地址。传递的是值的引用,也就是说传递前和传递后都指向同一个引用(也就是同一个内存空间)。
三、Java包
7.JDK 中常用的包有哪些📛
◾ java.lang:这个是系统的基础类;
◾ java.io:这里面是所有输入输出有关的类,比如文件操作等;
◾ java.nio:为了完善 io 包中的功能,提高 io 包中性能而写的一个新包;
◾ java.net:这里面是与网络有关的类;
◾ java.util :这个是系统辅助类,特别是集合类; . java.sql :这个是数据库操作的类。
8. import java 和javax有什么区别
. 刚开始的时候 JavaAPI 所必需的包是 java开头的包, javax 当时只是扩展 API 包来说使用。然而随 着时间的推移, javax 逐渐的扩展成为 Java API 的组成部分。但是,将扩展从 javax 包移动到 java 包将是太麻烦了,最终会破坏一堆现有的代码。因此,最终决定 javax 包将成为标准API的一部分。所以,实际上java和javax没有区别。这都是一个名字。