概述
不知道大家在平時(shí)的開發(fā)過程中或者源碼里是否留意過內(nèi)部類,那有思考過為什么要有內(nèi)部類,內(nèi)部類都有哪幾種形式,靜態(tài)內(nèi)部類和普通內(nèi)部類有什么區(qū)別呢?本篇文章主要帶領(lǐng)大家理解下這塊內(nèi)容。
內(nèi)部類介紹和分類
顧名思義,內(nèi)部類是指一個(gè)類在另外一個(gè)類的內(nèi)部,是定義在另一個(gè)類中的類。根據(jù)類的位置和屬性不同,可以分為下面幾種。
常規(guī)內(nèi)部類
@DatapublicclassTree{privateString treeName;privateString treeType;privateList leafs;@DatapublicclassLeaf{privateString color;privateString leafSize;publicvoidshow(){? ? ? ? ? ? System.out.println("tree name: "+ Tree.this.treeName);? ? ? ? ? ? System.out.println("tree name: "+ treeName);? ? ? ? }? ? }publicstaticvoidmain(String[] args){? ? ? ? Tree tree =newTree();? ? ? ? Leaf leaf = tree.new Leaf();? ? }}復(fù)制代碼
上面就是一個(gè)內(nèi)部類的例子,Leaf是Tree的內(nèi)部類。
Tree.this.treeName
Tree tree =newTree();Leaf leaf = tree.newLeaf();復(fù)制代碼
字節(jié)碼分析:
我們查看下內(nèi)部類的字節(jié)碼,如下圖:

我們看到內(nèi)部類Leaf的構(gòu)造方法init,實(shí)際上會隱式地將外部對象傳入,初始化,這樣才能在內(nèi)部類中訪問。
局部內(nèi)部類
局部內(nèi)部類,比如只有在一個(gè)方法內(nèi)部創(chuàng)建。
@DatapublicclassTree{privateString treeName;privateString treeType;privateList leafs;publicvoidtreeRoot(intheight){classTreeRoot{privateintrootHeight;publicvoidshowRootHeight(){this.rootHeight = height;? ? ? ? ? ? ? ? System.out.println("root height "+ height);? ? ? ? ? ? }? ? ? ? }? ? ? ? TreeRoot root =newTreeRoot();? ? ? ? root.showRootHeight();? ? }}復(fù)制代碼
局部類不能用public或private訪問說明符進(jìn)行聲明,因?yàn)樽饔糜蛑辉谶@個(gè)局部,沒有必要。
局部內(nèi)部類可以完全將自己隱藏起來,體現(xiàn)良好的封裝性。
局部內(nèi)部類可以直接訪問方法的變量。
匿名內(nèi)部類
匿名內(nèi)部類,就更加簡潔了,連類名都省略了, 這個(gè)配合lambda食用,非常簡便,想必大家也經(jīng)常使用了。
publicstaticvoidmain(String[] args){? ? ? ? Runnable runnable =newRunnable() {? ? ? ? ? ? @Overridepublicvoidrun(){? ? ? ? ? ? ? ? System.out.println("runnable");? ? ? ? ? ? }? ? ? ? };? ? ? ? Runnable runnable2 = () -> System.out.println("runnable");? ? }復(fù)制代碼
匿名內(nèi)部類也體現(xiàn)了良好的封裝性和簡潔性。
靜態(tài)內(nèi)部類
最后,再介紹下靜態(tài)內(nèi)部類,這個(gè)也使用的非常頻繁。java源碼中也有很多這樣的例子,比如HashMap中,Node節(jié)點(diǎn)就是一個(gè)靜態(tài)內(nèi)部類。

如果用static來修飾一個(gè)內(nèi)部類,那么就是靜態(tài)內(nèi)部類。這個(gè)內(nèi)部類屬于外部類本身,但是不屬于外部類的任何對象。因此使用static修飾的內(nèi)部類稱為靜態(tài)內(nèi)部類。靜態(tài)內(nèi)部類有如下規(guī)則:
靜態(tài)內(nèi)部類不能訪問外部類的實(shí)例成員,只能訪問外部類的類成員。
外部類可以使用靜態(tài)內(nèi)部類的類名作為調(diào)用者來訪問靜態(tài)內(nèi)部類的類成員,也可以使用靜態(tài)內(nèi)部類對象訪問其實(shí)例成員。
靜態(tài)內(nèi)部類和普通內(nèi)部類的區(qū)別
我們可以通俗按下面的方式理解:
內(nèi)部類:就是我是你的一部分,我了解你,我知道你的全部,沒有你就沒有我。(所以內(nèi)部類對象是以外部類對象存在為前提的)
靜態(tài)內(nèi)部類:就是我跟你沒關(guān)系,自己可以完全獨(dú)立存在,但是我就借你的殼用一下,來隱藏自己。
從字節(jié)碼角度上看,靜態(tài)內(nèi)部類和非靜態(tài)內(nèi)部類最大的區(qū)別是:非靜態(tài)內(nèi)部類編譯后隱式保存著外部類的引用(就算外部類對象沒用了也GC不掉),但是靜態(tài)內(nèi)部類沒有。
內(nèi)部類的作用
通過上面的了解,我想內(nèi)部類還是有一定的意義的。
內(nèi)部類方法可以訪問該類定義所在的作用域中的數(shù)據(jù),包括私有的數(shù)據(jù)。
內(nèi)部類可以對同一個(gè)包中的其他類隱藏起來。
其實(shí)內(nèi)部類更多的時(shí)候,不想把這個(gè)類暴露出去,它可能只是外部類的一個(gè)邏輯組成部分,不需要其他地方知道什么,這時(shí)候我們用內(nèi)部類更加的清楚。