大家好,我是閑者,最近正在考慮找新工作,進(jìn)行面試,但是工作時間比較久了,很多基礎(chǔ)知識都很模糊,所以得復(fù)習(xí)下,順便做下記錄,也便于大家參考。
以下為大綱,后期會定期更新
2023最新Java面試題(一) - Java 基礎(chǔ)
2023最新Java面試題(七) - 網(wǎng)絡(luò)
2023最新Java面試題(十) - Spring Boot/Spring Cloud
2023最新Java面試題(十一) - Hibernate(擴展)
2023最新Java面試題(十五) - Zookeeper(擴展)
當(dāng)面試涉及Java基礎(chǔ)時,通常會涉及Java語言的核心概念、特性和一些常見的編程場景。以下是一些常見的Java基礎(chǔ)面試題以及相應(yīng)的答案。
一. Java 基礎(chǔ)
1. Java的特點是什么?
Java是一種面向?qū)ο蟮木幊陶Z言,具有跨平臺性、強類型、自動內(nèi)存管理(垃圾回收)、多線程支持等特點。
2. JDK 和 JRE 有什么區(qū)別?
JDK:Java Development Kit 的簡稱,Java 開發(fā)工具包,提供了 Java 的開發(fā)環(huán)境和運行環(huán)境。
JRE:Java Runtime Environment 的簡稱,Java 運行環(huán)境,為 Java 的運行提供了所需環(huán)境。
具體來說 JDK 其實包含了 JRE,同時還包含了編譯 Java 源碼的編譯器 Javac,還包含了很多 Java 程序調(diào)試和分析的工具。簡單來說:如果你需要運行 Java 程序,只需安裝 JRE 就可以了,如果你需要編寫 Java 程序,需要安裝 JDK。
3. == 和 equals 的區(qū)別是什么?
「== 解讀」
對于基本類型和引用類型 == 的作用效果是不同的,如下所示:
基本類型:比較的是值是否相同;
引用類型:比較的是引用是否相同;
代碼示例:
String x = "string";
String y = "string";
String z = new String("string");
System.out.println(x==y); // true
System.out.println(x==z); // false
System.out.println(x.equals(y)); // true
System.out.println(x.equals(z)); // true
代碼解讀:因為 x 和 y 指向的是同一個引用,所以 == 也是 true,而 new String()方法則重寫開辟了內(nèi)存空間,所以 == 結(jié)果為 false,而 equals 比較的一直是值,所以結(jié)果都為 true。
「equals 解讀」
equals 本質(zhì)上就是 ==,只不過 String 和 Integer 等重寫了 equals 方法,把它變成了值比較??聪旅娴拇a就明白了。
首先來看默認(rèn)情況下 equals 比較一個有相同值的對象,代碼如下:
class Cat{
public Cat(String name){
this.name = name;
}
private String name;
public String getName(){
return name;
}
public void setName(String name){
this.name = name;
}
}
Cat c1 = new Cat("王磊");
Cat c2 = new Cat("王磊");
System.out.println(c1.equals(c2)); // false
輸出結(jié)果出乎我們的意料,竟然是 false?這是怎么回事,看了 equals 源碼就知道了,源碼如下:
public boolean equals(Object obj){
return (this == obj);
}
原來 equals 本質(zhì)上就是 ==。
那問題來了,兩個相同值的 String 對象,為什么返回的是 true?代碼如下:
String s1 = new String("老王");
String s2 = new String("老王");
System.out.println(s1.equals(s2)); // true
同樣的,當(dāng)我們進(jìn)入 String 的 equals 方法,找到了答案,代碼如下:
public boolean equals(Object anObject){
if (this == anObject) {
return true;
}
if (anObject instanceof String) {
String anotherString = (String)anObject;
int n = value.length;
if (n == anotherString.value.length) {
char v1[] = value;
char v2[] = anotherString.value;
int i = 0;
while (n-- != 0) {
if (v1[i] != v2[i])
return false;
i++;
}
return true;
}
}
return false;
}
原來是 String 重寫了 Object 的 equals 方法,把引用比較改成了值比較。
「總結(jié)」 :== 對于基本類型來說是值比較,對于引用類型來說是比較的是引用;而 equals 默認(rèn)情況下是引用比較,只是很多類重新了 equals 方法,比如 String、Integer 等把它變成了值比較,所以一般情況下 equals 比較的是值是否相等。
4. 兩個對象的 hashCode() 相同,則 equals() 也一定為 true,對嗎?
不對,兩個對象的 hashCode() 相同,equals() 不一定 true。
代碼示例:
String str1 = "通話";
String str2 = "重地";
System.out.println(String.format("str1:%d | str2:%d", str1.hashCode(), str2.hashCode()));
System.out.println(str1.equals(str2));
執(zhí)行的結(jié)果:
str1:1179395 | str2:1179395 false
代碼解讀:很顯然“通話”和“重地”的 hashCode() 相同,然而 equals() 則為 false,因為在散列表中,hashCode() 相等即兩個鍵值對的哈希值相等,然而哈希值相等,并不一定能得出鍵值對相等。
5. final 在 Java 中有什么作用?
final 修飾的類叫最終類,該類不能被繼承。
final 修飾的方法不能被重寫。
final 修飾的變量叫常量,常量必須初始化,初始化之后值就不能被修改。
6. Java 中的 Math. round(-1. 5) 等于多少?
等于 -1,因為在數(shù)軸上取值時,中間值(0.5)向右取整,所以正 0.5 是往上取整,負(fù) 0.5 是直接舍棄。
7. String 屬于基礎(chǔ)的數(shù)據(jù)類型嗎?
String 不屬于基礎(chǔ)類型,基礎(chǔ)類型有 8 種:
byte、boolean、char、short、int、float、long、double
而 String 屬于對象。
8. Java 中操作字符串都有哪些類?它們之間有什么區(qū)別?
操作字符串的類有:String、StringBuffer、StringBuilder。
String 和 StringBuffer、StringBuilder 的區(qū)別在于 String 聲明的是不可變的對象,每次操作都會生成新的 String 對象,然后將指針指向新的 String 對象,而 StringBuffer、StringBuilder 可以在原有對象的基礎(chǔ)上進(jìn)行操作,所以在經(jīng)常改變字符串內(nèi)容的情況下最好不要使用 String。
StringBuffer 和 StringBuilder 最大的區(qū)別在于,StringBuffer 是線程安全的,而 StringBuilder 是非線程安全的,但 StringBuilder 的性能卻高于 StringBuffer,所以在單線程環(huán)境下推薦使用 StringBuilder,多線程環(huán)境下推薦使用 StringBuffer。
9. String str="i"與 String str=new String("i")一樣嗎?
不一樣,因為內(nèi)存的分配方式不一樣。String str="i"的方式,Java 虛擬機會將其分配到常量池中;而 String str=new String("i") 則會被分到堆內(nèi)存中。
10. 如何將字符串反轉(zhuǎn)?
使用 StringBuilder 或者 stringBuffer 的 reverse() 方法。
示例代碼:
// StringBuffer reverse
StringBuffer stringBuffer = new StringBuffer();
stringBuffer.append("abcdefg");
System.out.println(stringBuffer.reverse()); // gfedcba
// StringBuilder reverse
StringBuilder stringBuilder = new StringBuilder();
stringBuilder.append("abcdefg");
System.out.println(stringBuilder.reverse()); // gfedcba
11. String 類的常用方法都有那些?
indexOf():返回指定字符的索引。
charAt():返回指定索引處的字符。
replace():字符串替換。
trim():去除字符串兩端空白。
split():分割字符串,返回一個分割后的字符串?dāng)?shù)組。
getBytes():返回字符串的 byte 類型數(shù)組。
length():返回字符串長度。
toLowerCase():將字符串轉(zhuǎn)成小寫字母。
toUpperCase():將字符串轉(zhuǎn)成大寫字符。
substring():截取字符串。
equals():字符串比較。
12. 抽象類必須要有抽象方法嗎?
不需要,抽象類不一定非要有抽象方法。
示例代碼:
abstract classCat{
publicstaticvoidsayHi(){
System. out. println("hi~");
}
}
上面代碼,抽象類并沒有抽象方法但完全可以正常運行。
13. 普通類和抽象類有哪些區(qū)別?
普通類不能包含抽象方法,抽象類可以包含抽象方法。
抽象類不能直接實例化,普通類可以直接實例化。
14. 抽象類能使用 final 修飾嗎?
不能,定義抽象類就是讓其他類繼承的,如果定義為 final 該類就不能被繼承,這樣彼此就會產(chǎn)生矛盾,所以 final 不能修飾抽象類,編輯器也會提示錯誤信息:Illegal combination of modifiers: 'abstract' and 'final'
15. 接口和抽象類有什么區(qū)別?
實現(xiàn):抽象類的子類使用 extends 來繼承;接口必須使用 implements 來實現(xiàn)接口。
構(gòu)造函數(shù):抽象類可以有構(gòu)造函數(shù);接口不能有。
實現(xiàn)數(shù)量:類可以實現(xiàn)很多個接口;但是只能繼承一個抽象類。
訪問修飾符:接口中的方法默認(rèn)使用 public 修飾;抽象類中的方法可以是任意訪問修飾符。
16. Java 中 IO 流分為幾種?
按功能來分:輸入流(input)、輸出流(output)。
按類型來分:字節(jié)流和字符流。
字節(jié)流和字符流的區(qū)別是:字節(jié)流按 8 位傳輸以字節(jié)為單位輸入輸出數(shù)據(jù),字符流按 16 位傳輸以字符為單位輸入輸出數(shù)據(jù)。
17. BIO、NIO、AIO 有什么區(qū)別?
BIO:Block IO 同步阻塞式 IO,就是我們平常使用的傳統(tǒng) IO,它的特點是模式簡單使用方便,并發(fā)處理能力低。
NIO:Non IO 同步非阻塞 IO,是傳統(tǒng) IO 的升級,客戶端和服務(wù)器端通過 Channel(通道)通訊,實現(xiàn)了多路復(fù)用。
AIO:Asynchronous IO 是 NIO 的升級,也叫 NIO2,實現(xiàn)了異步非堵塞 IO ,異步 IO 的操作基于事件和回調(diào)機制。
18. Files的常用方法都有哪些?
Files.exists():檢測文件路徑是否存在。
Files.createFile():創(chuàng)建文件。
Files.createDirectory():創(chuàng)建文件夾。
Files.delete():刪除一個文件或目錄。
Files.copy():復(fù)制文件。
Files.move():移動文件。
Files.size():查看文件個數(shù)。
Files.read():讀取文件。
Files.write():寫入文件。
19. 什么是JVM(Java虛擬機)?
JVM是Java程序的運行環(huán)境,它負(fù)責(zé)將Java源代碼編譯成字節(jié)碼并執(zhí)行。JVM提供內(nèi)存管理和垃圾回收,以確保程序的可靠性和性能。
20. Java中的基本數(shù)據(jù)類型有哪些?
Java的基本數(shù)據(jù)類型包括整數(shù)類型(int, byte, short, long)、浮點類型(float, double)、字符類型(char)、布爾類型(boolean)。
21. 什么是自動裝箱和拆箱?
自動裝箱是將基本數(shù)據(jù)類型自動轉(zhuǎn)換為其相應(yīng)的包裝類型,而拆箱是將包裝類型自動轉(zhuǎn)換為基本數(shù)據(jù)類型。例如,int到Integer的轉(zhuǎn)換是自動裝箱,Integer到int的轉(zhuǎn)換是自動拆箱。
22. 請解釋Java中的多態(tài)性。
多態(tài)性是指一個對象可以以多種形式存在。在Java中,多態(tài)性通常通過繼承和接口實現(xiàn),允許不同類的對象對同一方法做出不同的響應(yīng)。
23. 什么是Java中的異常處理機制?
異常處理是一種用于處理運行時錯誤的機制。在Java中,異常可以通過try-catch-finally塊來捕獲和處理,也可以通過throws關(guān)鍵字聲明在方法簽名中。
24. 什么是Java的訪問修飾符?
Java的訪問修飾符用于控制類、變量、方法和構(gòu)造函數(shù)的可見性和訪問權(quán)限。常見的訪問修飾符包括public、private、protected和默認(rèn)(無修飾符)。
25. 什么是Java中的繼承?
繼承是一種面向?qū)ο缶幊痰母拍?,它允許一個類(子類/派生類)繼承另一個類(父類/基類)的屬性和方法。子類可以通過繼承來重用父類的代碼,并可以添加新的功能。
26. 什么是接口和抽象類的區(qū)別?
接口是一種純粹的抽象類,其中所有方法都是抽象的,并且不包含成員變量。抽象類可以包含非抽象方法和成員變量。一個類可以實現(xiàn)多個接口,但只能繼承一個抽象類。
27. 請解釋Java中的多線程編程。
多線程編程允許在同一時間執(zhí)行多個線程,每個線程獨立執(zhí)行。Java提供了多線程支持,通過Thread類或Runnable接口創(chuàng)建和管理線程。多線程可以提高程序的并發(fā)性和性能。
這些問題涵蓋了Java基礎(chǔ)知識的一些重要方面。在面試中,可能還會涉及到更具體的主題,例如集合框架、反射、Lambda表達(dá)式等。為了成功通過Java基礎(chǔ)面試,建議深入學(xué)習(xí)這些核心概念并進(jìn)行練習(xí)。