主頁 > 後端開發 > 萬字長文,62道Java核心面試題,一次性打包送給積極向上的你

萬字長文,62道Java核心面試題,一次性打包送給積極向上的你

2020-09-30 23:10:30 後端開發

先看再點贊,給自己一點思考的時間,微信搜索【沉默王二】關注這個靠才華茍且的程式員,
本文 GitHub github.com/itwanger 已收錄,里面還有一線大廠整理的面試題,以及我的系列文章,

二哥,你好,找作業找了仨月,還沒有找到,很焦慮,我該怎么辦呢?你那有沒有 Java 方面的面試題可以分享一波啊?

以上是讀者田田給我發的私信,看完后于我心有戚戚焉啊,最近境況確實不容樂觀,并非是個人的原因造成的,那,既然需要面試題,二哥就義不容辭,必須得準備一波,

這次我花了兩周的時間,準備了 62 道 Java 核心面試題,希望能夠幫助到田田,以及其他和田田類似情況的讀者朋友,

01、請說出 Java 14 版本中更新的重要功能

Java 14 發布于 2020 年 3 月 17 日,更新的重要功能有:

  • switch 運算式
  • instanceof 增強運算式,預覽功能
  • 文本塊,第二次預覽
  • Records,預覽功能

剛好我之前寫過一篇文章,關于 Java 14 的開箱體驗,很香,讀者朋友需要的話,可以點下面的鏈接看一看,

Java 14 開箱,它真香香香香

02、請說出 Java 13 版本中更新的重要功能

Java 13 發布于 2019 年 9 月 17 日,更新的重要功能有:

  • 文本塊,預覽功能
  • switch 運算式,預覽功能
  • Java Socket 重新實作
  • FileSystems.newFileSystem() 方法
  • 支持 Unicode 12.1
  • 可伸縮、低延遲的垃圾收集器改進,用于回傳未使用的記憶體

03、請說出 Java 12 版本中更新的重要功能

Java 12 發布于 2019 年 3 月 19 日,更新的重要功能有:

  • JVM 更新
  • File.mismatch() 方法
  • 緊湊型數字格式
  • String 類新增了一些方法,比如說 indent()

04、請說出 Java 11 版本中更新的重要功能

Java 11 是繼 Java 8 之后的第二個商用版本,如果你下載的是 Oracle JDK,則需要進行付費;如果想繼續使用免費版本,需要下載 Open JDK,

Oracle JDK 中會有一些 Open JDK 沒有的、商用閉源的功能,

Java 11 更新的重要功能有:

  • 可以直接使用 java 命令運行 Java 程式,源代碼將會隱式編譯和運行,
  • String 類新增了一些方法,比如說 isBlank()lines()strip() 等等,
  • Files 類新增了兩個讀寫方法,readString()writeString()
  • 可以在 Lambda 運算式中使用 var 作為變數型別,

05、請說出 Java 10 版本中更新的重要功能

Java 10 更新的重要功能有:

  • 區域變數型別推斷,舉個例子,var list = new ArrayList();,可以使用 var 來作為變數型別,Java 編譯器知道 list 的型別為字串的 ArrayList,
  • 增強 java.util.Locale
  • 提供了一組默認的根證書頒發機構(CA),

06、請說出 Java 9 版本中更新的重要功能

Java 9 更新的重要功能有:

  • 模塊系統
  • 不可變的 List、Set、Map 的工廠方法
  • 介面中可以有私有方法
  • 垃圾收集器改進

07、請說出 Java 8 版本中更新的重要功能

Java 8 發布于 2014 年 3 月份,可以說是 Java 6 之后最重要的版本更新,深受開發者的喜愛,

  • 函式式編程和 Lambda 運算式
  • Stream 流
  • Java Date Time API
  • 介面中可以使用默認方法和靜態方法

我強烈建議點開上面的鏈接閱讀以下,以正確理解這些概念,

08、請說出 Java 面向物件編程中的一些重要概念

  • 抽象
  • 封裝
  • 多型
  • 繼承

09、Java 聲稱的平臺獨立性指的是什么?

常見的作業系統有 Windows、Linux、OS-X,那么平臺獨立性意味著我們可以在任何作業系統中運行相同源代碼的 Java 程式,比如說我們可以在 Windows 上撰寫 Java 程式,然后在 Linux 上運行它,

10、什么是 JVM?

JVM(Java Virtual Machine)俗稱 Java 虛擬機,之所以稱為虛擬機,是因為它實際上并不存在,它提供了一種運行環境,可供 Java 位元組碼在上面運行,

JVM 提供了以下操作:

  • 加載位元組碼
  • 驗證位元組碼
  • 執行位元組碼
  • 提供運行時環境

JVM 定義了以下內容:

  • 存盤區
  • 類檔案格式
  • 暫存器組
  • 垃圾回收堆
  • 致命錯誤報告等

我們來嘗試理解一下 JVM 的內部結構,它包含了類加載器(Class Loader)、運行時資料區(Runtime Data Areas)和執行引擎(Excution Engine),

1)類加載器

類加載器是 JVM 的一個子系統,用于加載類檔案,每當我們運行一個 Java 程式,它都會由類加載器首先加載,Java 中有三個內置的類加載器:

  • 啟動類加載器(Bootstrap Class-Loader),加載 jre/lib 包下面的 jar 檔案,比如說常見的 rt.jar(包含了 Java 標準庫下的所有類檔案,比如說 java.lang 包下的類,java.net 包下的類,java.util 包下的類,java.io 包下的類,java.sql 包下的類),

  • 擴展類加載器(Extension or Ext Class-Loader),加載 jre/lib/ext 包下面的 jar 檔案,

  • 應用類加載器(Application or App Clas-Loader),根據程式的類路徑(classpath)來加載 Java 類,

一般來說,Java 程式員并不需要直接同類加載器進行互動,JVM 默認的行為就已經足夠滿足大多數情況的需求了,不過,如果遇到了需要和類加載器進行互動的情況,而對類加載器的機制又不是很了解的話,就不得不花大量的時間去除錯
ClassNotFoundExceptionNoClassDefFoundError 等例外,

對于任意一個類,都需要由它的類加載器和這個類本身一同確定其在 JVM 中的唯一性,也就是說,如果兩個類的加載器不同,即使兩個類來源于同一個位元組碼檔案,那這兩個類就必定不相等(比如兩個類的 Class 物件不 equals),

是不是有點暈,來來來,通過一段簡單的代碼了解下,

public class Test {

    public static void main(String[] args) {
        ClassLoader loader = Test.class.getClassLoader();
        while (loader != null) {
            System.out.println(loader.toString());
            loader = loader.getParent();
        }
    }

}

每個 Java 類都維護著一個指向定義它的類加載器的參考,通過 類名.class.getClassLoader() 可以獲取到此參考;然后通過 loader.getParent() 可以獲取類加載器的上層類加載器,

上面這段代碼的輸出結果如下:

sun.misc.Launcher$AppClassLoader@18b4aac2
sun.misc.Launcher$ExtClassLoader@4617c264

第一行輸出為 Test 的類加載器,即應用類加載器,它是 sun.misc.Launcher$AppClassLoader 類的實體;第二行輸出為擴展類加載器,是 sun.misc.Launcher$ExtClassLoader 類的實體,那啟動類加載器呢?

按理說,擴展類加載器的上層類加載器是啟動類加載器,但在我這個版本的 JDK 中, 擴展類加載器的 getParent() 回傳 null,所以沒有輸出,

2)運行時資料區

運行時資料區又包含以下內容,

  • PC暫存器(PC Register),也叫程式計數器(Program Counter Register),是一塊較小的記憶體空間,它的作用可以看做是當前執行緒所執行的位元組碼的信號指示器,

  • JVM 堆疊(Java Virtual Machine Stack),與 PC 暫存器一樣,JVM 堆疊也是執行緒私有的,每一個 JVM 執行緒都有自己的 JVM 堆疊,這個堆疊與執行緒同時創建,它的生命周期與執行緒相同,

  • 本地方法堆疊(Native Method Stack),JVM 可能會使用到傳統的堆疊來支持 Native 方法(使用 Java 語言以外的其它語言[C語言]撰寫的方法)的執行,這個堆疊就是本地方法堆疊,

  • 堆(Heap),在 JVM 中,堆是可供各條執行緒共享的運行時記憶體區域,也是供所有類實體和資料物件分配記憶體的區域,

  • 方法區(Method area),在 JVM 中,被加載型別的資訊都保存在方法區中,包括型別資訊(Type Information)和方法串列(Method Tables),方法區是所有執行緒共享的,所以訪問方法區資訊的方法必須是執行緒安全的,

  • 運行時常量池(Runtime Constant Pool),運行時常量池是每一個類或介面的常量池在運行時的表現形式,它包括了編譯器可知的數值字面量,以及運行期決議后才能獲得的方法或欄位的參考,簡而言之,當一個方法或者變數被參考時,JVM 通過運行時常量區來查找方法或者變數在記憶體里的實際地址,

3)執行引擎

執行引擎包含了:

  • 解釋器:讀取位元組碼流,然后執行指令,因為它一條一條地解釋和執行指令,所以它可以很快地解釋位元組碼,但是執行起來會比較慢,

  • 即時(Just-In-Time,JIT)編譯器:即時編譯器用來彌補解釋器的缺點,提高性能,執行引擎首先按照解釋執行的方式來執行,然后在合適的時候,即時編譯器把整段位元組碼編譯成本地代碼,然后,執行引擎就沒有必要再去解釋執行方法了,它可以直接通過本地代碼去執行,執行本地代碼比一條一條進行解釋執行的速度快很多,編譯后的代碼可以執行的很快,因為本地代碼是保存在快取里的,

11、JDK 和 JVM 有什么區別?

JDK 是 Java Development Kit 的首字母縮寫,是提供給 Java 開發人員的軟體環境,包含 JRE 和一組開發工具,可分為以下版本:

  • 標準版(大多數開發人員用的就是這個)
  • 企業版
  • 微型版

JDK 包含了一個私有的 JVM 和一些其他資源,比如說編譯器(javac 命令)、解釋器(java 命令)等,幫助 Java 程式員完成開發作業,

12、JVM 和 JRE 有什么區別?

Java Runtime Environment(JRE)是 JVM 的實作,JRE 由 JVM 和 Java 二進制檔案以及其他類組成,可以執行任何程式,JRE 不包含 Java 編譯器,除錯器等任何開發工具,

13、哪個類是所有類的超類?

java.lang.Object 是所有 Java 類的超類,我們不需要繼承它,因為是隱式繼承的,

14、為什么 Java 不支持多重繼承?

如果有兩個類共同繼承(extends)一個有特定方法的父類,那么該方法會被兩個子類重寫,然后,如果你決定同時繼承這兩個子類,那么在你呼叫該重寫方法時,編譯器不能識別你要呼叫哪個子類的方法,這也正是著名的菱形問題,見下圖,

ClassC 同時繼承了 ClassA 和 ClassB,ClassC 的物件在呼叫 ClassA 和 ClassB 中多載的方法時,就不知道該呼叫 ClassA 的方法,還是 ClassB 的方法,

15、為什么 Java 不是純粹的面向物件編程語言?

之所以不能說 Java 是純粹的面向物件編程語言,是因為 Java 支持基本資料型別,比如說 int、short、long、double 等,盡管它們有自己的包裝器型別,但它們的確不能算是物件,

16、path 和 classpath 之間有什么區別?

path 是作業系統用來查找可執行檔案的環境變數,我的電腦上就定義了下圖這些 path 變數,比如 Java 和 Maven 的,

classpath 是針對 Java 而言的,用于指定 Java 虛擬機載入的位元組碼檔案路徑,

17、Java 中 `main()` 方法的重要性是什么?

每個程式都需要一個入口,對于 Java 程式來說,入口就是 main 方法,

public static void main(String[] args) {}
  • public 關鍵字是另外一個訪問修飾符,除了可以宣告方法和變數(所有類可見),還可以宣告類,main() 方法必須宣告為 public,

  • static 關鍵字表示該變數或方法是靜態變數或靜態方法,可以直接通過類訪問,不需要實體化物件來訪問,

  • void 關鍵字用于指定方法沒有回傳值,

另外,main 關鍵字為方法的名字,Java 虛擬機在執行程式時會尋找這個識別符號;args 為 main() 方法的引數名,它的型別為一個 String 陣列,也就是說,在使用 java 命令執行程式的時候,可以給 main() 方法傳遞字串陣列作為引數,

java HelloWorld 沉默王二 沉默王三

javac 命令用來編譯程式,java 命令用來執行程式,HelloWorld 為這段程式的類名,沉默王二和沉默王三為字串陣列,中間通過空格隔開,然后就可以在 main() 方法中通過 args[0]args[1] 獲取傳遞的引數值了,

public class HelloWorld {
    public static void main(String[] args) {
        if ("沉默王二".equals(args[0])) {

        }

        if ("沉默王三".equals(args[1])) {

        }
    }
}

main() 方法的寫法并不是唯一的,還有其他幾種變體,盡管它們可能并不常見,可以簡單來了解一下,

第二種,把方括號 [] 往 args 靠近而不是 String 靠近:

public static void main(String []args) { }

第三種,把方括號 [] 放在 args 的右側:

public static void main(String args[]) { }

第四種,還可以把陣列形式換成可變引數的形式:

public static void main(String...args) { }

第五種,在 main() 方法上添加另外一個修飾符 strictfp,用于強調在處理浮點數時的兼容性:

public strictfp static void main(String[] args) { }

也可以在 main() 方法上添加 final 關鍵字或者 synchronized 關鍵字,

第六種,還可以為 args 引數添加 final 關鍵字:

public static void main(final String[] args) { }

第七種,最復雜的一種,所有可以添加的關鍵字統統添加上:

final static synchronized strictfp void main(final String[] args) { }

當然了,并不需要為了裝逼特意把 main() 方法寫成上面提到的這些形式,使用 IDE 提供的默認形式就可以了,

18、Java 的重寫(Override)和多載(Overload)有什么區別?

先來看一段重寫的代碼吧,

class LaoWang{
    public void write() {
        System.out.println("老王寫了一本《基督山伯爵》");
    }
}
public class XiaoWang extends LaoWang {
    @Override
    public void write() {
        System.out.println("小王寫了一本《茶花女》");
    }
}

重寫的兩個方法名相同,方法引數的個數也相同;不過一個方法在父類中,另外一個在子類中,就好像父類 LaoWang 有一個 write() 方法(無參),方法體是寫一本《基督山伯爵》;子類 XiaoWang 重寫了父類的 write() 方法(無參),但方法體是寫一本《茶花女》,

來寫一段測驗代碼,

public class OverridingTest {
    public static void main(String[] args) {
        LaoWang wang = new XiaoWang();
        wang.write();
    }
}

大家猜結果是什么?

小王寫了一本《茶花女》

在上面的代碼中,們宣告了一個型別為 LaoWang 的變數 wang,在編譯期間,編譯器會檢查 LaoWang 類是否包含了 write() 方法,發現 LaoWang 類有,于是編譯通過,在運行期間,new 了一個 XiaoWang 物件,并將其賦值給 wang,此時 Java 虛擬機知道 wang 參考的是 XiaoWang 物件,所以呼叫的是子類 XiaoWang 中的 write() 方法而不是父類 LaoWang 中的 write() 方法,因此輸出結果為“小王寫了一本《茶花女》”,

再來看一段多載的代碼吧,

class LaoWang{
    public void read() {
        System.out.println("老王讀了一本《Web全堆疊開發進階之路》");
    }

    public void read(String bookname) {
        System.out.println("老王讀了一本《" + bookname + "》");
    }
}

多載的兩個方法名相同,但方法引數的個數不同,另外也不涉及到繼承,兩個方法在同一個類中,就好像類 LaoWang 有兩個方法,名字都是 read(),但一個有引數(書名),另外一個沒有(只能讀寫死的一本書),

來寫一段測驗代碼,

public class OverloadingTest {
    public static void main(String[] args) {
        LaoWang wang = new LaoWang();
        wang.read();
        wang.read("金瓶");
    }
}

這結果就不用猜了,變數 wang 的型別為 LaoWang,wang.read() 呼叫的是無參的 read() 方法,因此先輸出“老王讀了一本《Web全堆疊開發進階之路》”;wang.read("金瓶") 呼叫的是有參的 read(bookname) 方法,因此后輸出“老王讀了一本《金瓶》”,在編譯期間,編譯器就知道這兩個 read() 方法時不同的,因為它們的方法簽名(=方法名稱+方法引數)不同,

簡單的來總結一下:

1)編譯器無法決定呼叫哪個重寫的方法,因為只從變數的型別上是無法做出判斷的,要在運行時才能決定;但編譯器可以明確地知道該呼叫哪個多載的方法,因為參考型別是確定的,引數個數決定了該呼叫哪個方法,

2)多型針對的是重寫,而不是多載,

  • 如果在一個類中有多個相同名字的方法,但引數不同,則稱為方法多載,

  • 父類中有一個方法,子類中有另外一個和它有相同簽名(方法名相同,引數相同、修飾符相同)的方法時,則稱為方法重寫,子類在重寫父類方法的時候可以加一個 @Override 注解,

19、`main()` 方法可以多載嗎?

可以,一個類中可以有多個名稱為“main”的方法:

public class MainTest {
    public static void main(String[] args) {
        System.out.println("main(String[] args)");
    }

    public static void main(String[] args,String arg) {
        System.out.println("(String[] args,String arg");
    }
}

但該類在運行的時候,只會找到一個入口,即 public static void main(String[] args)

20、一個 Java 源檔案中有多個 public 類嗎?

一個 Java 源檔案中不能有多個 public 類,

21、什么是 Java 的 package(包)?

在 Java 中,我們使用 package(包)對相關的類、介面和子包進行分組,這樣做的好處有:

  • 使相關型別更容易查找
  • 避免命名沖突,比如說 com.itwanger.Hello 和 com.itwangsan.Hello 不同
  • 通過包和訪問權限控制符來限定類的可見性

可以使用 package 關鍵字來定義一個包名,需要注意的是,這行代碼必須處于一個類中的第一行,強烈建議在包中宣告類,不要預設,否則就失去了包結構的帶來的好處,

包的命名應該遵守以下規則:

  • 應該全部是小寫字母
  • 可以包含多個單詞,單詞之間使用“.”連接,比如說 java.lang
  • 名稱由公司名或者組織名確定,采用倒序的方式,比如說,我個人博客的域名是 www.itwanger.com,所以我創建的包名是就是 com.itwanger.xxxx

每個包或者子包都在磁盤上有自己的目錄結構,如果 Java 檔案時在 com.itwanger.xxxx 包下,那么該檔案所在的目錄結構就應該是 com->itwanger->xxxx

默認情況下,java.lang 包是默認匯入的,我們不需要顯式地匯入該包下的任何類,

package com.cmower.bb;

public class PackageTest {
    public static void main(String[] args) {
        Boolean.toString(true);
    }
}

Boolean 類屬于 java.lang 包,當使用它的時候并不需要顯式匯入,

22、什么是訪問權限修飾符?

訪問權限修飾符對于 Java 來說,非常重要,目前共有四種:public、private、protected 和 default(預設),

一個類只能使用 public 或者 default 修飾,public 修飾的類你之前已經見到過了,現在我來定義一個預設權限修飾符的類給你欣賞一下,

class Dog {
}

哈哈,其實也沒啥可以欣賞的,預設意味著這個類可以被同一個包下的其他類進行訪問;而 public 意味著這個類可以被所有包下的類進行訪問,

假如硬要通過 private 和 protected 來修飾類的話,編譯器會生氣的,它不同意,

private 可以用來修飾類的構造方法、欄位和方法,只能被當前類進行訪問,protected 也可以用來修飾類的構造方法、欄位和方法,但它的權限范圍更寬一些,可以被同一個包中的類進行訪問,或者當前類的子類,

可以通過下面這張圖來對比一下四個權限修飾符之間的差別:

  • 同一個類中,不管是哪種權限修飾符,都可以訪問;
  • 同一個包下,private 修飾的無法訪問;
  • 子類可以訪問 public 和 protected 修飾的;
  • public 修飾符面向世界,哈哈,可以被所有的地方訪問到,

23、什么是 final 關鍵字?

final 關鍵字修飾類的時候,表示該類無法被繼承,比如,String 類就是 final 的,無法被繼承,

final 關鍵字修飾方法的時候,表示子類無法覆寫它,

final 關鍵字修飾變數的時候,表示該變數只能被賦值一次,盡管變數的狀態可以更改,

關于 final 更詳細的內容,可以參照我之前寫了另外一篇文章:

我去,你竟然還不會用 final 關鍵字

24、什么是 static 關鍵字?

static 關鍵字可以用來修飾類變數,使其具有全域性,即所有物件將共享同一個變數,

static 關鍵字可以用來修飾方法,該方法稱為靜態方法,只可以訪問類的靜態變數,并且只能呼叫類的靜態方法,

關于 static 更詳細的內容,可以參照我之前寫了另外一篇文章:

面試官:兄弟,說說Java的static關鍵字吧

25、finally 和 finalize 有什么區別?

finally 通常與 try-catch 塊一起使用,即使 try-catch 塊引發了例外,finally 塊中的代碼也會被執行,用于釋放 try 塊中創建的資源,

finalize() 是 Object 類的一個特殊方法,當物件正在被垃圾回收時,垃圾收集器將會呼叫該方法,可以重寫該方法用于釋放系統資源,

26、可以將一個類宣告為 static 的嗎?

不能將一個外部類宣告為 static 的,但可以將一個內部類宣告為 static 的——稱為靜態內部類,

27、什么是靜態匯入?

如果必須在一個類中使用其他類的靜態變數或者靜態方法,通常我們需要先匯入該類,然后使用“類名.變數/方法”的形式呼叫,

import java.lang.Math;

double test = Math.PI * 5;

也可以通過靜態匯入的方式,就不需要再使用類名了,

import static java.lang.Math.PI;

double test = PI * 5;

不過,靜態匯入容易引發混亂(變數名或者方法名容易沖突),因此最好避免使用靜態匯入,

28、什么是 try-with-resources?

try-with-resources 是 Java 7 時引入的一個自動資源管理陳述句,在此之前,我們必須通過 try-catch-finally 的方式手動關閉資源,當我們忘記關閉資源的時候,就容易導致記憶體泄漏,

關于 try-with-resources 更詳細的內容,可以參照我之前寫了另外一篇文章:

我去,你竟然還在用 try–catch-finally

29、什么是 multi-catch?

Java 7 改進的另外一個地方就是 multi-catch,可以在單個 catch 中捕獲多個例外,當一個 try 塊拋出多個類似的例外時,這種寫法更短,更清晰,

catch(IOException | SQLException ex){
     logger.error(ex);
     throw new MyException(ex.getMessage());
}

當有多個例外的時候,可以使用管道表示符“|”隔開,

30、什么是 static 塊?

static 塊是由 Java ClassLoader 將類加載到記憶體中時執行的代碼塊,通常用于初始化類的靜態變數或者創建靜態資源,

31、什么是介面?

介面是 Java 編程語言中的一個核心概念,不僅在 JDK 原始碼中使用很多,還在 Java 設計模式、框架和工具中使用很多,介面提供了一種在 Java 中實作抽象的方法,用于定義子類的行為約定,

關于介面更詳細的內容,可以參照我之前寫了另外一篇文章:

可能是把 Java 介面講得最通俗的一篇文章

32、什么是抽象類?

在 Java 中,抽象類用于創建具有某些被子類實作的默認方法的類,一個抽象類可以有沒有方法體的抽象方法,也可以有和普通類一樣有方法體的方法,

abstract 關鍵字用于宣告一個抽象類,抽象類無法實體化,主要用于為子類提供一個模板,子類需要覆寫抽象方法,

關于抽象類更詳細的內容,可以參照我之前寫了另外一篇文章:

小白,你要的Java抽象類,操碎了心

33、抽象類和介面有什么區別?

  • 宣告抽象類的關鍵字為 abstract,宣告介面的關鍵字為 interface,
  • 抽象類可以有具體的方法,介面不能,
  • 一個類只能繼承一個抽象類,但可以實作多個介面,
  • 介面中的變數只能是隱式的常量,抽象類中可以有任意型別的變數,
  • 如果一個抽象類有 main() 方法,則可以運行它;但介面不能,
  • 抽象類是對類的一種抽象,繼承抽象類的類和抽象類本身是一種 is-a 的關系,
  • 介面是對類的某種行為的一種抽象,介面和類之間并沒有很強的關聯關系,所有的類都可以實作 Serializable 介面,從而具有序列化的功能,

34、一個介面可以實作或者繼承另外一個介面嗎?

介面不能實作另外一個介面,但可以繼承一個介面,

因為介面中不能有具體的方法,所以不會出現菱形問題,所以我們可以在一個介面中繼承多個介面,

public interface C extends A,B{
}

從 Java 8 開始,介面可以有默認方法,所以當多個介面中存在相同的默認方法時,需要在實作介面的類中提供該方法的實作,

35、什么是標記介面?

標記介面是一個空的介面,沒有任何方法,用于強制實作類中的某些功能,比較出名的標記介面有 Serializable 介面、Cloneable 介面,

關于 Serializable 介面更詳細的內容,可以參照我之前寫了另外一篇文章:

Java Serializable:明明就一個空的介面嘛

36、什么是包裝器類?

包裝器類是 Java 中八種基本資料型別的物件表示形式,所有的包裝器類都是不可變的,并且是 final 的,通過裝箱和拆箱,可以將八種基本資料型別和包裝器型別互相轉換,

關于基本型別和包裝型別更詳細的內容,可以參照我之前寫了另外一篇文章:

面試官:兄弟,說說基本型別和包裝型別的區別吧

37、什么是列舉?

enum(列舉)是 Java 1.5 時引入的關鍵字,它表示一種特殊型別的類,默認繼承自 java.lang.Enum,

public enum PlayerType {
    TENNIS,
    FOOTBALL,
    BASKETBALL
}

enum 是用于創建列舉的關鍵字,列舉中的常量都是隱式 static 和 final 的,

關于列舉更詳細的內容,可以參照我之前寫了另外一篇文章:

恕我直言,我懷疑你并不會用 Java 列舉

38、什么是 Java 注解?

注解是 Java 1.5 時引入的,同 class 和 interface 一樣,也屬于一種型別,注解提供了一系列資料用來裝飾程式代碼(類、方法、欄位等),但是注解并不是所裝飾代碼的一部分,它對代碼的運行效果沒有直接影響(這句話怎么理解呢?),由編譯器決定該執行哪些操作,

關于注解更詳細的內容,可以參照我之前寫了另外一篇文章:

不吹牛逼,擼個注解有什么難的

39、什么是 Java 反射?

Java 反射機制是在運行狀態中,對于任意一個類,都能夠知道這個類的所有欄位和方法;對于任意一個物件,都能夠呼叫它的任意欄位和方法;這種動態獲取資訊以及動態呼叫物件方法的功能稱為 Java 反射機制,

反射屬于高級主題,在常規編程中應該避免使用,因為反射可以通過呼叫私有的構造方法來破壞設計模式,比如說單例模式,

盡管不建議使用反射機制,但反射機制的存在至關重要,因為如果沒有反射,我們將沒有 Spring 之類的框架,甚至 Tomcat 之類的服務器,它們通過反射呼叫適當的方法并對類實體化,省去了很多麻煩,

40、Java 中的組合指的什么?

通過物件組合可以實作代碼的重用,Java 組合是通過參考其他物件的參考來實作的,使用組合的好處就是我們可以控制其他物件對使用者的可見性,并且刻意重用我們需要的物件,

41、與繼承相比,組合有什么好處?

  • 任何父類的修改都可能會影響到子類,甚至我們沒有使用父類的一些方法,舉個例子,假如子類有一個方法 test(),而父類之前是沒有的,但突然有人在不知情的情況下在父類插入了一個同名但簽名不同的 test() 方法,那么就會出現編譯錯誤,組合是不會遇到這個問題的,因為我們僅僅會使用我們需要的方法,

這是父類追加的 test() 方法:

public class Super {
    public String test() {
        System.out.println("super");
        return null;
    }
}

原來子類的 test() 方法就出錯了,

來個表格列舉一下兩者之間的優缺點:

組 合 關 系 繼 承 關 系
優點:不破壞封裝,整體類與區域類之間松耦合,彼此相對獨立 缺點:破壞封裝,子類與父類之間緊密耦合,子類依賴于父類的實作,子類缺乏獨立性
優點:具有較好的可擴展性 缺點:支持擴展,但是往往以增加系統結構的復雜度為代價
優點:支持動態組合,在運行時,整體物件可以選擇不同型別的區域物件 缺點:不支持動態繼承,在運行時,子類無法選擇不同的父類
優點:整體類可以對區域類進行包裝,封裝區域類的介面,提供新的介面 缺點:子類不能改變父類的介面
缺點:整體類不能自動獲得和區域類同樣的介面 優點:子類能自動繼承父類的介面
缺點:創建整體類的物件時,需要創建所有區域類的物件 優點:創建子類的物件時,無須創建父類的物件

42、如何在 Java 中對自定義物件的集合進行排序?

需要對自定義物件的類實作 Comparable 介面,重寫 compareTo(T obj) 方法,該方法在排序的時候會被呼叫進行排序,

關于 Comparable 和 Comparator 介面更詳細的內容,可以參照我之前寫了另外一篇文章:

一文徹底搞懂Java中的Comparable和Comparator

43、什么是內部類?

我們可以在一個類中定義一個類,這個類被稱為內部類,內部類可以訪問外部類的所有變數和方法,內部類中不能有任何靜態變數,

44、什么是匿名內部類?

沒有名稱的內部類稱為匿名內部類,它通過單個陳述句進行定義和實體化,總是需要擴展一個類或者實作一個介面,

由于匿名內部類沒有名稱,所以無法為匿名內部類定義構造方法,

45、什么是 Java 中的 Classloader(類加載器)?

當我們要訪問任何類時,都需要通過 Java Classloader 將該類的位元組碼加在到記憶體當中,可以通過繼承 ClassLoader 并重寫 loadClass(String name) 方法來創建自定義的類加載器,

46、有哪些不同的類加載器?

  • Bootstrap 類加載器,用來加在 JDK 的內部類,比如說 rt.jar,

  • Extensions 類加載器,它從 JDK 擴展目錄(JAVA_HOME/lib/ext)中加載類,

  • System 類加載器,它從當前類路徑加載類,

47、什么是三元運算子?

三元運算子是 if-then-else 陳述句的一個替換,示例如下:

result = testStatement ? value1 : value2;

48、super 關鍵字有什么用?

當在子類中重寫了父類方法時,可以通過 super 關鍵字訪問父類方法,

也可以使用 super 關鍵字在子類構造方法中呼叫父類構造方法,它必須是構造方法中的第一條陳述句,

public class SuperClass {

    public SuperClass(){
    }

    public SuperClass(int i){}

    public void test(){
        System.out.println("父類的測驗方法");
    }
}

來看子類中如何使用 super 關鍵字:

public class ChildClass extends SuperClass {

    public ChildClass(String str){
        // 呼叫父類的構造方法
        super();

        // 呼叫子類的 test 方法
        test();

        // 使用 super 關鍵字呼叫父類的 test 方法
        super.test();
    }

    @Override
    public void test(){
        System.out.println("child class test method");
    }
}

49、什么是 break,什么是 continue?

我們可以使用 break 關鍵字終止 for、while、do-while 回圈;可以在 switch 陳述句中使用 break 退出 case 條件,

我們可以使用 continue 關鍵字在 for、while、do-while 回圈跳過當前迭代;甚至可以使用帶有標簽的 continue 陳述句來跳過最外層回圈的當前迭代,

50、什么是 this 關鍵字?

this 關鍵字提供對當前物件的參考,主要用于確保使用了當前物件的變數,而不是具有相同名稱的區域變數,

//constructor
public Point(int x, int y) {
    this.x = x;
    this.y = y;
}

還可以使用 this 關鍵字在構造方法中呼叫其他構造方法:

public Rectangle() {
    this(0000);
}
public Rectangle(int width, int height) {
    this(00, width, height);
}
public Rectangle(int x, int y, int width, int height) {
    this.x = x;
    this.y = y;
    this.width = width;
    this.height = height;
}

關于 this 關鍵字更詳細的內容,可以參照我之前寫了另外一篇文章:

我去,你竟然還不會用 this 關鍵字

51、什么是默認的構造方法?

一個類的無參構造方法被稱為默認構造方法,當我們沒有為一個類定義構造方法時,Java 編譯器會自動為該類創建一個默認的無參構造方法,如果定義了其他構造方法,編譯器就不會在為我們創建默認構造方法了,

52、try 塊可以沒有 catch 嗎?

是的,可以直接使用 try-finally,而不需要 catch 捕獲例外,

53、什么是垃圾回收?

垃圾回收(Garbage Collection,簡稱 GC)會查看堆記憶體,識別正在使用和未使用的物件,以及會自動洗掉未使用的物件,用來釋放記憶體,

54、什么是序列化和反序列化?

我們可以把一個 Java 物件轉化成一個資料流,這被稱為序列化,一旦物件被轉化為資料流后,就可以將其保存到檔案或者通過網路套接字發送,

如果一個物件實作了 Serializable 介面,就可以使用 java.io.ObjectOutputStream 將物件寫入檔案,

將資料流再轉化為 Java 物件被稱為反序列化,

55、如何通過命令提示符運行 jar 檔案?

可以通過 java 命令運行 jar 檔案,但需要 jar 檔案中有 main 方法,

56、System 類有什么用?

System 類是 Java 的一個核心類,比較常用的就是 System.out.println()

System 類是 final 的,因此我們不能通過繼承來重寫它的方法,System 類沒有提供任何 public 的構造方法,因此無法實體化,它的所有方法都是 static 的,

57、什么是 instanceof 關鍵字?

我們可以使用 instanceof 關鍵字檢查物件是否屬于一個類,

public static void main(String args[]){
    Object str = new String("沉默王二");

    if(str instanceof String){
        System.out.println("字串值為:" + str);
    }

    if(str instanceof Integer){
        System.out.println("數字的值是:" + str);
    }
}

58、可以在 switch 中使用字串嗎?

Java 7 改進的一個功能就是允許在 switch 陳述句中使用字串,

關于 switch 更詳細的內容,可以參照我之前寫了另外一篇文章:

我去,你寫的 switch 陳述句也太老土了吧

59、Java 是按值傳遞還是按參考傳遞?

可以很確定地說,Java 是按值傳遞的,

關于這個問題,可以參照我之前寫了另外一篇文章:

面試官:兄弟,說說Java到底是值傳遞還是參考傳遞

60、堆(heap)和堆疊(stack)有什么區別?

  • 堆記憶體被應用程式的所有部分使用,而堆疊記憶體僅由執行執行緒使用,
  • 當我們創建物件時,它始終存盤在堆空間上;堆疊僅存盤該物件的參考,堆疊記憶體還可以存盤區域的基本型別資料變數,
  • 堆疊的記憶體管理方式是 LIFO(后進先出)形式,而堆的記憶體管理方式更復雜,

61、Java 編譯在 JDK 中,還是 JRE,還是 JVM 中?

Java 編譯器的任務是將 Java 源代碼轉換為位元組碼,可以通過 javac 命令執行,因此它在 JDK 中,JRE 中不需要它,

62、下面這段程式輸出什么?

public class Test {

    public static String toString(){
        System.out.println("測驗 toString 方法有沒有被呼叫");
        return "";
    }

    public static void main(String args[]){
        System.out.println(toString());
    }
}

這段代碼無法編譯通過,因為 java.lang.Object 中的 toString() 方法不是 static 的,它無法被 static 修飾的方法重寫,

那下面這段代碼呢?

public class Test {
    public static String foo(){
        System.out.println("測驗 foo 方法有沒有被呼叫");
        return "";
    }

    public static void main(String args[]){
        Test obj = null;
        System.out.println(obj.foo());
    }
}

這段代碼會輸出 測驗 foo 方法有沒有被呼叫,沒有出現 NullPointerException,為什么呢?

命名 obj 為 null 啊,通過 obj 呼叫 foo() 方法的時候應該拋出 NullPointerException 例外才對啊!

之所以沒有拋出例外,是因為 Java 編譯器對這段代碼做出了優化,因為 foo() 方法是靜態方法,所以 obj.foo() 會被優化為 foo(),所以就不會拋出例外了,

來看一下這段代碼的位元組碼就明白了:

public class Test {
    public Test() {
    }

    public static String foo() {
        System.out.println("測驗 foo 方法有沒有被呼叫");
        return "";
    }

    public static void main(String[] args) {
        Test obj = null;
        System.out.println(foo());
    }
}

鳴謝

說句實在話,這 62 道 Java 核心面試題在面試的程序中還是很常見的,值得好好復習一遍,

好了,我親愛的小伙伴們,能看到這里絕逼是最優秀的程式員,二哥必須伸出帥氣的大拇指為你點個贊!


我是沉默王二,一枚有顏值卻靠才華茍且的程式員,關注即可提升學習效率,別忘了三連啊,點贊、收藏、留言,我不挑,奧利給

注:如果文章有任何問題,歡迎毫不留情地指正,

很多讀者都同情我說,“二哥母豬似的日更原創累不累?”我只能說寫作不易,且行且珍惜啊,歡迎微信搜索「沉默王二」第一時間閱讀,回復「簡歷」更有我精心為你準備的簡歷模板,本文 GitHub github.com/itwanger 已收錄,歡迎 star,

轉載請註明出處,本文鏈接:https://www.uj5u.com/houduan/143462.html

標籤:Java

上一篇:Spring Boot整合swagger使用教程

下一篇:Spring 如何解決回圈依賴問題?

標籤雲
其他(157675) Python(38076) JavaScript(25376) Java(17977) C(15215) 區塊鏈(8255) C#(7972) AI(7469) 爪哇(7425) MySQL(7132) html(6777) 基礎類(6313) sql(6102) 熊猫(6058) PHP(5869) 数组(5741) R(5409) Linux(5327) 反应(5209) 腳本語言(PerlPython)(5129) 非技術區(4971) Android(4554) 数据框(4311) css(4259) 节点.js(4032) C語言(3288) json(3245) 列表(3129) 扑(3119) C++語言(3117) 安卓(2998) 打字稿(2995) VBA(2789) Java相關(2746) 疑難問題(2699) 细绳(2522) 單片機工控(2479) iOS(2429) ASP.NET(2402) MongoDB(2323) 麻木的(2285) 正则表达式(2254) 字典(2211) 循环(2198) 迅速(2185) 擅长(2169) 镖(2155) 功能(1967) .NET技术(1958) Web開發(1951) python-3.x(1918) HtmlCss(1915) 弹簧靴(1913) C++(1909) xml(1889) PostgreSQL(1872) .NETCore(1853) 谷歌表格(1846) Unity3D(1843) for循环(1842)

熱門瀏覽
  • 【C++】Microsoft C++、C 和匯編程式檔案

    ......

    uj5u.com 2020-09-10 00:57:23 more
  • 例外宣告

    相比于斷言適用于排除邏輯上不可能存在的狀態,例外通常是用于邏輯上可能發生的錯誤。 例外宣告 Item 1:當函式不可能拋出例外或不能接受拋出例外時,使用noexcept 理由 如果不打算拋出例外的話,程式就會認為無法處理這種錯誤,并且應當盡早終止,如此可以有效地阻止例外的傳播與擴散。 示例 //不可 ......

    uj5u.com 2020-09-10 00:57:27 more
  • Codeforces 1400E Clear the Multiset(貪心 + 分治)

    鏈接:https://codeforces.com/problemset/problem/1400/E 來源:Codeforces 思路:給你一個陣列,現在你可以進行兩種操作,操作1:將一段沒有 0 的區間進行減一的操作,操作2:將 i 位置上的元素歸零。最終問:將這個陣列的全部元素歸零后操作的最少 ......

    uj5u.com 2020-09-10 00:57:30 more
  • UVA11610 【Reverse Prime】

    本人看到此題沒有翻譯,就附帶了一個自己的翻譯版本 思考 這一題,它的第一個要求是找出所有 $7$ 位反向質數及其質因數的個數。 我們應該需要質數篩篩選1~$10^{7}$的所有數,這里就不慢慢介紹了。但是,重讀題,我們突然發現反向質數都是 $7$ 位,而將它反過來后的數字卻是 $6$ 位數,這就說明 ......

    uj5u.com 2020-09-10 00:57:36 more
  • 統計區間素數數量

    1 #pragma GCC optimize(2) 2 #include <bits/stdc++.h> 3 using namespace std; 4 bool isprime[1000000010]; 5 vector<int> prime; 6 inline int getlist(int ......

    uj5u.com 2020-09-10 00:57:47 more
  • C/C++編程筆記:C++中的 const 變數詳解,教你正確認識const用法

    1、C中的const 1、區域const變數存放在堆疊區中,會分配記憶體(也就是說可以通過地址間接修改變數的值)。測驗代碼如下: 運行結果: 2、全域const變數存放在只讀資料段(不能通過地址修改,會發生寫入錯誤), 默認為外部聯編,可以給其他源檔案使用(需要用extern關鍵字修飾) 運行結果: ......

    uj5u.com 2020-09-10 00:58:04 more
  • 【C++犯錯記錄】VS2019 MFC添加資源不懂如何修改資源宏ID

    1. 首先在資源視圖中,添加資源 2. 點擊新添加的資源,復制自動生成的ID 3. 在解決方案資源管理器中找到Resource.h檔案,編輯,使用整個專案搜索和替換的方式快速替換 宏宣告 4. Ctrl+Shift+F 全域搜索,點擊查找全部,然后逐個替換 5. 為什么使用搜索替換而不使用屬性視窗直 ......

    uj5u.com 2020-09-10 00:59:11 more
  • 【C++犯錯記錄】VS2019 MFC不懂的批量添加資源

    1. 打開資源頭檔案Resource.h,在其中預先定義好宏 ID(不清楚其實ID值應該設定多少,可以先新建一個相同的資源項,再在這個資源的ID值的基礎上遞增即可) 2. 在資源視圖中選中專案資源,按F7編輯資源檔案,按 ID 型別 相對路徑的形式添加 資源。(別忘了先把檔案拷貝到專案中的res檔案 ......

    uj5u.com 2020-09-10 01:00:19 more
  • C/C++編程筆記:關于C++的參考型別,專供新手入門使用

    今天要講的是C++中我最喜歡的一個用法——參考,也叫別名。 參考就是給一個變數名取一個變數名,方便我們間接地使用這個變數。我們可以給一個變數創建N個參考,這N + 1個變數共享了同一塊記憶體區域。(參考型別的變數會占用記憶體空間,占用的記憶體空間的大小和指標型別的大小是相同的。雖然參考是一個物件的別名,但 ......

    uj5u.com 2020-09-10 01:00:22 more
  • 【C/C++編程筆記】從頭開始學習C ++:初學者完整指南

    眾所周知,C ++的學習曲線陡峭,但是花時間學習這種語言將為您的職業帶來奇跡,并使您與其他開發人員區分開。您會更輕松地學習新語言,形成真正的解決問題的技能,并在編程的基礎上打下堅實的基礎。 C ++將幫助您養成良好的編程習慣(即清晰一致的編碼風格,在撰寫代碼時注釋代碼,并限制類內部的可見性),并且由 ......

    uj5u.com 2020-09-10 01:00:41 more
最新发布
  • Rust中的智能指標:Box<T> Rc<T> Arc<T> Cell<T> RefCell<T> Weak

    Rust中的智能指標是什么 智能指標(smart pointers)是一類資料結構,是擁有資料所有權和額外功能的指標。是指標的進一步發展 指標(pointer)是一個包含記憶體地址的變數的通用概念。這個地址參考,或 ” 指向”(points at)一些其 他資料 。參考以 & 符號為標志并借用了他們所 ......

    uj5u.com 2023-04-20 07:24:10 more
  • Java的值傳遞和參考傳遞

    值傳遞不會改變本身,參考傳遞(如果傳遞的值需要實體化到堆里)如果發生修改了會改變本身。 1.基本資料型別都是值傳遞 package com.example.basic; public class Test { public static void main(String[] args) { int ......

    uj5u.com 2023-04-20 07:24:04 more
  • [2]SpinalHDL教程——Scala簡單入門

    第一個 Scala 程式 shell里面輸入 $ scala scala> 1 + 1 res0: Int = 2 scala> println("Hello World!") Hello World! 檔案形式 object HelloWorld { /* 這是我的第一個 Scala 程式 * 以 ......

    uj5u.com 2023-04-20 07:23:58 more
  • 理解函式指標和回呼函式

    理解 函式指標 指向函式的指標。比如: 理解函式指標的偽代碼 void (*p)(int type, char *data); // 定義一個函式指標p void func(int type, char *data); // 宣告一個函式func p = func; // 將指標p指向函式func ......

    uj5u.com 2023-04-20 07:23:52 more
  • Django筆記二十五之資料庫函式之日期函式

    本文首發于公眾號:Hunter后端 原文鏈接:Django筆記二十五之資料庫函式之日期函式 日期函式主要介紹兩個大類,Extract() 和 Trunc() Extract() 函式作用是提取日期,比如我們可以提取一個日期欄位的年份,月份,日等資料 Trunc() 的作用則是截取,比如 2022-0 ......

    uj5u.com 2023-04-20 07:23:45 more
  • 一天吃透JVM面試八股文

    什么是JVM? JVM,全稱Java Virtual Machine(Java虛擬機),是通過在實際的計算機上仿真模擬各種計算機功能來實作的。由一套位元組碼指令集、一組暫存器、一個堆疊、一個垃圾回收堆和一個存盤方法域等組成。JVM屏蔽了與作業系統平臺相關的資訊,使得Java程式只需要生成在Java虛擬機 ......

    uj5u.com 2023-04-20 07:23:31 more
  • 使用Java接入小程式訂閱訊息!

    更新完微信服務號的模板訊息之后,我又趕緊把微信小程式的訂閱訊息給實作了!之前我一直以為微信小程式也是要企業才能申請,沒想到小程式個人就能申請。 訊息推送平臺🔥推送下發【郵件】【短信】【微信服務號】【微信小程式】【企業微信】【釘釘】等訊息型別。 https://gitee.com/zhongfuch ......

    uj5u.com 2023-04-20 07:22:59 more
  • java -- 緩沖流、轉換流、序列化流

    緩沖流 緩沖流, 也叫高效流, 按照資料型別分類: 位元組緩沖流:BufferedInputStream,BufferedOutputStream 字符緩沖流:BufferedReader,BufferedWriter 緩沖流的基本原理,是在創建流物件時,會創建一個內置的默認大小的緩沖區陣列,通過緩沖 ......

    uj5u.com 2023-04-20 07:22:49 more
  • Java-SpringBoot-Range請求頭設定實作視頻分段傳輸

    老實說,人太懶了,現在基本都不喜歡寫筆記了,但是網上有關Range請求頭的文章都太水了 下面是抄的一段StackOverflow的代碼...自己大修改過的,寫的注釋挺全的,應該直接看得懂,就不解釋了 寫的不好...只是希望能給視頻網站開發的新手一點點幫助吧. 業務場景:視頻分段傳輸、視頻多段傳輸(理 ......

    uj5u.com 2023-04-20 07:22:42 more
  • Windows 10開發教程_編程入門自學教程_菜鳥教程-免費教程分享

    教程簡介 Windows 10開發入門教程 - 從簡單的步驟了解Windows 10開發,從基本到高級概念,包括簡介,UWP,第一個應用程式,商店,XAML控制元件,資料系結,XAML性能,自適應設計,自適應UI,自適應代碼,檔案管理,SQLite資料庫,應用程式到應用程式通信,應用程式本地化,應用程式 ......

    uj5u.com 2023-04-20 07:22:35 more