詳解Java內(nèi)部類——匿名內(nèi)部類
今天來看看另一個更加神奇的類——匿名內(nèi)部類。
就像它的名字表示的那樣,這個類是匿名的,用完之后,深藏功與名,就像掃地僧那樣默默潛藏于深山之中。匿名內(nèi)部類不僅沒有名字,連class關(guān)鍵字都省掉了,而且匿名內(nèi)部類必須繼承于某個類或者實現(xiàn)某個接口,長的就像這樣:
new 父類(參數(shù)列表)|實現(xiàn)接口() { //匿名內(nèi)部類的內(nèi)部定義 }
來看一個栗子:
public abstract class Human { public abstract void walk(); }
這是一個抽象類,如果使用匿名內(nèi)部類來繼承的話是這樣的:
public class AnonymousTest { public static void main(String[] args) { Human human = new Human(){ public void walk(){ System.out.println("AnonymousHuman can walk."); }; }; human.walk(); } }
簡單粗暴,看起來就像局部內(nèi)部類的簡化版。如果不使用匿名內(nèi)部類,會是怎樣呢?
我們需要先創(chuàng)建一個類來繼承這抽象類:
public class Man extends Human { @Override public void walk() { System.out.println("Man can walk."); } }
然后再來使用這個類:
public class AnonymousTest { public static void main(String[] args) { Human human = new Man(); human.walk(); } }
因為一個單獨(dú)的類往往放在一個單獨(dú)的文件中,如果這個類只需要創(chuàng)建一個對象,那未免有些大材小用了,從上面的栗子可以比較出匿名內(nèi)部類的一個優(yōu)勢:在類只需要創(chuàng)建一個對象的情況下更加簡單方便。
再舉一個實際一點(diǎn)的栗子:
public class AnonymousTest { public static void main(String[] args) { Thread t = new Thread() { public void run() { for (int i = 0; i < 10; i++) { try { sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(i); } } }; t.start(); } }
這里創(chuàng)建了一個繼承于Thread的匿名內(nèi)部類,覆蓋了其中的 run方法,并創(chuàng)建了一個實例返回給了t,然后再調(diào)用run方法,可以看到,匿名內(nèi)部類只能存在一個實例對象,因為new過一次就無法再創(chuàng)建了,也許會覺得局部內(nèi)部類已經(jīng)很局限了,為什么要出現(xiàn)比局部內(nèi)部類適用范圍更小的匿名內(nèi)部類?、
這你就不懂了吧,在Java的實際使用中,匿名內(nèi)部類大有用處,為什么要使用匿名內(nèi)部類呢?
有時候,我們創(chuàng)建的類只需要一個實例,比如說在多線程中,要使用多線程,一般先繼承Thread類或者實現(xiàn)Runnable接口,然后再去調(diào)用它的方法,而每個任務(wù)一般都不一樣,每次都新建一個類顯然會很難管理,因為每個類只用一次就丟掉了,這個時候使用匿名內(nèi)部類就很方便了,不僅不需要管理一堆一次性類,而且創(chuàng)建起來簡單粗暴。就像上述栗子,還能簡化成這樣:
public class AnonymousTest { public static void main(String[] args) { new Thread() { public void run() { for (int i = 0; i < 10; i++) { try { sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(i); } } }.start(); } }
創(chuàng)建實例后直接調(diào)用run方法,簡單粗暴。
匿名內(nèi)部類不僅可以繼承于類,也可以實現(xiàn)于接口,比如說這樣:
public class AnonymousTest { public static void main(String[] args) { new Thread(new Runnable() { public void run() { for (int i = 0; i < 10; i++) { try { sleep(1000); } catch (InterruptedException e) { e.printStackTrace(); } System.out.println(i); } } }).start(); } }
當(dāng)然,還有些不得不用內(nèi)部類的情況,類只能繼承于一個類,如果一個類需要使用到另一個包中的另一個類的一個protected方法,卻已經(jīng)繼承于另一個類,那么這個時候就不得不用內(nèi)部類來解決了。
比如說,還有一個Woman(女人)類:
public class Woman { protected void dance(){ System.out.println("Woman can dance."); } }
這個時候,如果Man(男人)也難不住寂寞,想要dance(跳舞)一下,那該怎么辦呢?繼承Woman類?顯然不合乎邏輯,而且也無法實現(xiàn),因為已經(jīng)繼承于Human類了,但就是想要dance,該怎么辦?
內(nèi)部類的出現(xiàn)讓這個問題變得很簡單:
public class Man extends Human { @Override public void walk() { System.out.println("Man can walk."); } public void dance(){ new Woman(){ public void manDance(){ super.dance(); } }.manDance(); } }
因為在不同的包下,不能直接使用Woman的dance方法,但是可以用內(nèi)部類來繼承,從而調(diào)用protected方法,然后再放入Man的方法中,這樣,Man也能像Woman一樣dance了:
public class AnonymousTest { public static void main(String[] args) { Man human = new Man(); human.walk(); human.dance(); } }
當(dāng)然,使用匿名內(nèi)部類還是有很多限制的:
1、匿名內(nèi)部類必須是繼承一個類或者實現(xiàn)一個接口,但是兩者不可兼得,同時也只能繼承一個類或者實現(xiàn)一個接口。
2、匿名內(nèi)部類不能定義構(gòu)造函數(shù)。
3、匿名內(nèi)部類中不能存在任何的靜態(tài)成員變量和靜態(tài)方法。
4、匿名內(nèi)部類是特殊的局部內(nèi)部類,所以局部內(nèi)部類的所有限制同樣對匿名內(nèi)部類生效。
5、匿名內(nèi)部類不能是抽象的,它必須要實現(xiàn)繼承的類或者實現(xiàn)的接口的所有抽象方法。
那么問題來了,怎樣初始化一個匿名內(nèi)部類呢?畢竟匿名內(nèi)部類是不能有構(gòu)造器的。
當(dāng)然,首先,還是可以使用初始化塊來實現(xiàn)的,就像這樣:
public class AnonymousTest { public static void main(String[] args) { Human human = new Human() { private String name; { name = "human"; } @Override public void walk() { System.out.println(name + " walk."); } }; human.walk(); } }
但是這樣顯然就比較呆板,不夠靈活,無法接受外部參數(shù),那么怎樣靈活使用呢?不要心急,方法總比問題多,還是有辦法解決的:
public class AnonymousTest { public static void main(String[] args) { Human human = new AnonymousTest().getHumanInstance("Frank"); human.walk(); } public Human getHumanInstance(final String name){ return new Human() { private String nameA; { nameA = name; } @Override public void walk() { System.out.println(nameA + " walk."); } }; } }
這里利用初始化塊來對匿名內(nèi)部類進(jìn)行初始化,注意,如果匿名內(nèi)部類需要使用外部的參數(shù)或者變量,那么必須使用final修飾,因為內(nèi)部類使用的其實是參數(shù)的拷貝,并不是參數(shù)本身,為了更明顯的表明參數(shù)不可變,編譯器會要求使用final關(guān)鍵字來修飾需要使用的變量。
至此,匿名內(nèi)部類講解完畢,歡迎大家繼續(xù)關(guān)注!
以上就是詳解Java內(nèi)部類——匿名內(nèi)部類的詳細(xì)內(nèi)容,更多關(guān)于Java 匿名內(nèi)部類的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Spark學(xué)習(xí)筆記之Spark SQL的具體使用
這篇文章主要介紹了Spark學(xué)習(xí)筆記之Spark SQL的具體使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2019-06-06Java多線程并發(fā)生產(chǎn)者消費(fèi)者設(shè)計模式實例解析
這篇文章主要介紹了Java多線程并發(fā)生產(chǎn)者消費(fèi)者設(shè)計模式實例解析,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友可以參考下2020-03-03Java中JVM的雙親委派、內(nèi)存溢出、垃圾回收和調(diào)優(yōu)詳解
這篇文章主要介紹了Java中JVM的雙親委派、內(nèi)存溢出、垃圾回收和調(diào)優(yōu)詳解,類加載器是Java虛擬機(jī)(JVM)的一個重要組成部分,它的主要作用是將類的字節(jié)碼加載到內(nèi)存中,并生成對應(yīng)的Class對象,需要的朋友可以參考下2023-07-07SpringBoot使用SOFA-Lookout監(jiān)控的方法
本文介紹SpringBoot使用螞蟻金服SOFA-Lookout配合Prometheus進(jìn)行監(jiān)控,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2019-03-03Java枚舉_動力節(jié)點(diǎn)Java學(xué)院整理
enum 的全稱為 enumeration, 是 JDK 5 中引入的新特性,存放在 java.lang 包中。這篇文章給大家介紹Java枚舉相關(guān)知識,需要的的朋友參考下2017-04-04