你不知道的Java內部類

程序員小樂 發佈 2020-01-22T09:08:09+00:00

Anywhere, it is a good in the past, recall the number of times many, all would be light.

點擊上方 "程式設計師小樂"關注, 星標或置頂一起成長

每天凌晨00點00分,第一時間與你相約


每日英文

Anywhere, it is a good in the past, recall the number of times many, all would be light.

任何地方,再好的過去,回憶的次數多了,一切也就淡了。


每日掏心話

挖掘自己的潛力,樹立自己的信心,開拓自己的事業,揚長避短,循序漸進。


來自:六脈神劍 | 責編:樂樂

連結:juejin.im/post/5df0a84fe51d4557f544f7ac

程式設計師小樂(ID:study_tech)第 754 次推文 圖片來自 Pexels


往日回顧:如何更規範化編寫 Java 代碼


正文


看了很多源碼,都有用到內部類,但是自己以前在生產環境上,用的確實少,也有用過但是很少,所以今天就打算好好的把它從頭到尾的過一遍。

定義

可以將一個類的定義放在里另一個類的內部,這就是內部類,所謂的內部類的概念只是出現在編譯階段,對於jvm層是沒有內部類這個概念的。我們可以利用內部類來解決


  • 類的單繼承問題,外部類不能再繼承的類可以交給內部類繼承

  • 我們可以通過定義內部類來實現一個類私屬於一個類,實現更好的封裝性

  • 代碼優化:它需要更少的代碼


分類

內部類可以分為:


  • 靜態內部類。

  • 非靜態內部類。


非靜態內部類又可以分為:


  • 成員內部類。

  • 方法內部類。

  • 匿名內部類。


靜態內部類

我感覺這個是用的最多的,你比如說Redis的key的設計, 因為我們要中間拼接:號,所以用靜態內部類去組成不同的key是非常好的,這樣可以讓相同類型的key在同一個文件目錄下

靜態內部類的定義和普通的靜態變量或者靜態方法的定義方法是一樣的,使用static關鍵字,只不過這次static是修飾在class上的,一般而言,只有靜態內部類才允許使用static關鍵字修飾,普通類的定義是不能用static關鍵字修飾的,這一點需要注意一下。

下面定義一個靜態內部類:

public class Out {
private static String name;
private int age;

public static class In{
private int age;
public void sayHello(){

System.out.println("my name is : "+name);
//--編譯報錯---
//System.out.println("my age is :"+ age);
}
}
}

在上述代碼中,In這個類就是一個靜態內部類。我們說內部類是可以訪問外部類的私有欄位和私有方法的,對於靜態內部類,它遵循一致的原則,只能訪問外部類的靜態成員。

上述代碼中,外部類的非靜態私有欄位age在靜態內部類中是不允許訪問的,而靜態欄位name則是可訪問的。下面我們看,如何創建一個靜態內部類的實例對象。

public static void main(String [] args){

Out.In innerClass = new Out.In();
innerClass.sayHello();
}

使用場景,一般來說,對於和外部類聯繫緊密但是並不依賴於外部類實例的情況下,可以考慮定義成靜態內部類。下面我們看稍顯複雜的成員內部類。

成員內部類

我們說了,四種不同類型的內部類都各自有各自的使用場景,靜態內部類適合於那種和外部類關係密切但是並不依賴外部類實例的情況。但是對於需要和外部類實例相關聯的情況下,可以選擇將內部類定義成成員內部類。

以下代碼定義了一個簡單的成員內部類:

public class Out {
private String name;

public void showName(){
System.out.println("my name is : "+name);
}

public class In{
public void sayHello(){
System.out.println(name);

Out.this.showName();
}
}
}

以上定義了一個簡單的內部類In,我們的成員內部類可以直接訪問外部類的成員欄位和成員方法,因為它是關聯著一個外部類實例的。下面我們看看在外部是如何創建該內部類實例的。

public static void main(String [] args){
Out out = new Out();
out.setName("六脈神劍")
Out.In in = out.new In();
in.sayHello();
}

因為成員內部類是關聯著一個具體的外部類實例的,所以它的實例創建必然是由外部類實例來創建的。

對於實例的創建,我們只需要記住即可,成員內部類的實例創建需要關聯外部類實例對象,靜態內部類實例創建相對簡單。下面我們主要看看在編譯階段編譯器是如何保持內部類對外部類成員信息可訪問的。

使用場景,對於那種要高度依賴外部類實例的情況下,定義一個成員內部類則會顯的更加明智。

方法內部類

方法內部類,顧名思義,定義在一個方法內部的類。方法內部類相對而言要複雜一些,下面定義一個方法內部類:

public class Out {
private String name;

public void sayHello(){
class In{
public void showName(){
System.out.println("my name is : "+name);
}
}

In in = new In();
in.showName();
}
}

我們定義了一個類,在該類中又定義了一個方法sayHello,然而在該方法中我們定義了一個內部類,類In就是一個方法內部類。我們的方法內部類的生命周期不超過包含它的方法的生命周期,也就是說,方法內部類只能在方法中使用。所以在聲明的時候,任何的訪問修飾符都是沒有意義的,於是Java乾脆不允許使用任何的訪問修飾符修飾方法內部類。

其中還需要注意一點的是,定義和使用時兩回事,別看那一大串定義類的代碼,你實際想要使用該類,就必須new對象,而對於方法內部類而言,只能在方法內部new對象。這就是方法內部類的簡單介紹,下面我們看看其實現原理。

有關方法內部類的實現原理其實是和成員內部類差不太多的,也是在內部類初始化的時候為其傳入一個外部類實例,區別在哪呢?就在於方法內部類是定義在具體方法的內部的,所以該類除了可以通過傳入的外部實例訪問外部類中的欄位和方法,對於包含它的方法中被傳入的參數也會隨著外部類實例一起初始化給內部類。

毋庸置疑的是,方法內部類的封裝性比之前介紹的兩種都要完善。所以一般只有在需要高度封裝的時候才會將類定義成方法內部類。

匿名內部類

可能內部類的所有分類中,匿名內部類的名號是最大的,也是我們最常用到的,多見於函數式編程,lambda表達式等。下面我們重點看看這個匿名內部類。

匿名內部類就是沒有名字的內部類,在定義完成同時,實例也創建好了,常常和new關鍵字緊密結合。當然,它也不局限於類,也可以是接口,可以出現在任何位置。

下面我們定義一個匿名內部類:

如果您必須重寫類或接口的方法,則應該使用它。可以通過兩種方式創建Java匿名內部類

//首先定義一個普通類
public class Out {
private String name;

public void sayHello(){
System.out.println("my name is :" + name);
}
}


//定義和使用一個匿名內部類
public static void main(String [] args){
Out out = new Out(){
@Override
public void sayHello(){
System.out.println("my name is cyy");
}
public void showName(){
System.out.println("hello single");
}
};
out.sayHello();
}

從上述代碼中可以很顯然的讓我們看出來,我們的匿名內部類必定是要依託一個父類的,因為它是沒有名字的,無法用一個具體的類型來表示。所以匿名內部類往往都是通過繼承一個父類,重寫或者重新聲明一些成員來實現一個匿名內部類的定義。實際上還是利用了里式轉換原理。

其實在看了上述三種內部類的原理之後,反而覺得匿名內部類的實現較為簡單了。主要思路還是將內部類抽離出來,通過初始化傳入外部類的實例以達到對外部類所有成員的訪問。只是在匿名內部類中,被依託的父類不是他的外部類。

匿名內部類的主要特點在於,沒有名字,對象只能被使用一次,可以出現在任意位置。所以它的使用場景也是呼之欲出,對於一些對代碼簡潔度有所要求的情況下,可首選匿名內部類。

總結

以上完成了對四種內部類的簡單介紹,對於他們各自實現的原理也都已經介紹過了。其實大致相同,由於jvm對每個類都要求一個單獨的源碼文件,所以編譯階段就完成了分離的操作,但是在分離的過程中又要保持內部類和外部類之間的這種聯繫,於是編譯器添加了一些接口保持這種信息共享的結構。

使用內部類可以大大增加程序的封裝性,使得代碼整體簡潔度較高。

講完這個後面的函數式接口 引用就好講一點了

結尾

內部類就講那麼多,希望大家以後看源碼會輕鬆點,哈哈


歡迎在留言區留下你的觀點,一起討論提高。如果今天的文章讓你有新的啟發,學習能力的提升上有新的認識,歡迎轉發分享給更多人。


猜你還想看


阿里、騰訊、百度、華為、京東最新面試題匯集

Java中關於try、catch、finally中的細節分析,看了都說好!

Google 開源的依賴注入庫,比 Spring 更小更快!

Spring怎麼在一個事務中開啟另一個事務


關注「程式設計師小樂」,收看更多精彩內容
嘿,你在看嗎?


關鍵字: