概述
不知道大家在平時(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í)際上會(huì)隱式地將外部對(duì)象傳入鞠柄,初始化,這樣才能在內(nèi)部類中訪問嫉柴。
局部?jī)?nèi)部類
局部?jī)?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è)局部夯尽,沒有必要。
局部?jī)?nèi)部類可以完全將自己隱藏起來登馒,體現(xiàn)良好的封裝性匙握。
局部?jī)?nèi)部類可以直接訪問方法的變量。
匿名內(nèi)部類
匿名內(nèi)部類陈轿,就更加簡(jiǎn)潔了圈纺,連類名都省略了, 這個(gè)配合lambda食用,非常簡(jiǎn)便麦射,想必大家也經(jīng)常使用了蛾娶。
publicstaticvoidmain(String[] args){? ? ? ? Runnable runnable =newRunnable() {? ? ? ? ? ? @Overridepublicvoidrun(){? ? ? ? ? ? ? ? System.out.println("runnable");? ? ? ? ? ? }? ? ? ? };? ? ? ? Runnable runnable2 = () -> System.out.println("runnable");? ? }復(fù)制代碼
匿名內(nèi)部類也體現(xiàn)了良好的封裝性和簡(jiǎ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)部類屬于外部類本身,但是不屬于外部類的任何對(duì)象切距。因此使用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)部類對(duì)象訪問其實(shí)例成員。
靜態(tài)內(nèi)部類和普通內(nèi)部類的區(qū)別
我們可以通俗按下面的方式理解:
內(nèi)部類:就是我是你的一部分葡幸,我了解你最筒,我知道你的全部,沒有你就沒有我蔚叨。(所以內(nèi)部類對(duì)象是以外部類對(duì)象存在為前提的)
靜態(tài)內(nèi)部類:就是我跟你沒關(guān)系床蜘,自己可以完全獨(dú)立存在,但是我就借你的殼用一下蔑水,來隱藏自己邢锯。
從字節(jié)碼角度上看,靜態(tài)內(nèi)部類和非靜態(tài)內(nèi)部類最大的區(qū)別是:非靜態(tài)內(nèi)部類編譯后隱式保存著外部類的引用(就算外部類對(duì)象沒用了也GC不掉)搀别,但是靜態(tài)內(nèi)部類沒有丹擎。
內(nèi)部類的作用
通過上面的了解,我想內(nèi)部類還是有一定的意義的歇父。
內(nèi)部類方法可以訪問該類定義所在的作用域中的數(shù)據(jù)蒂培,包括私有的數(shù)據(jù)。
內(nèi)部類可以對(duì)同一個(gè)包中的其他類隱藏起來庶骄。
其實(shí)內(nèi)部類更多的時(shí)候毁渗,不想把這個(gè)類暴露出去,它可能只是外部類的一個(gè)邏輯組成部分单刁,不需要其他地方知道什么灸异,這時(shí)候我們用內(nèi)部類更加的清楚。