匿名類(Anonymous inner classes) 是不能有名稱的類,所以沒辦法引用它們。必須在建立時,作為new語句的一部分來聲明它們。
這就要採用另一種形式的new語句,如下所示:
new <類或介面> <類的主體>
這種形式的new語句聲明一個新的匿名類,它對一個給定的類進行擴充,或者實現一個給定的介面。它還建立那個類的一個新執行個體,並把它作為語句的結果而返回。要擴充的類和要實現的介面是new語句的運算元,後跟匿名類的主體。
假如匿名類對另一個類進行擴充,它的主體可以訪問類的成員、覆蓋它的方法等等,這和其他任何標準的類都是一樣的。假如匿名類實現了一個介面,它的主體必須實現介面的方法。
注意: 匿名類的聲明是在編譯時間進行的,執行個體化在運行時進行。這意味著for迴圈中的一個new語句會建立相同匿名類的幾個執行個體,而不是建立幾個不同匿名類的一個執行個體。
從技術上說,匿名類可被視為非靜態內部類,所以它們具有和方法內部聲明的非靜態內部類一樣的許可權和限制。
假如要執行的任務需要一個對象,但卻不值得建立全新的對象(原因可能是所需的類過於簡單,或者是由於它只在一個方法內部使用),匿名類就顯得非常有用。匿名類尤其適合在Swing應用程式中快速建立事件處理常式.
在Java中,匿名類多用來處理事件(event handle)。但其實,它們對於debug也很有協助。下面將介紹如何利用匿名類來簡化你的debug。
我們該如何調試那些非自己源碼的方法調用呢?比方說,對Jbutton.setEnable()的調用。Java提供的匿名類,可以很好的解決這個問題。
通常,當我們繼承一個類時,我們可以通過提供新的方法來覆蓋(override)該類中現有的方法:
public class MyButton extends JButton
{
public void setVisible( boolean visible )
{
// Rolling our own visibility
}
}
在執行個體化(instantiate)MyButton類之後,任何對方法setVisible()的調用,都會調用上面代碼中的 setVisible()方法。可問題是,我們不想僅僅為了覆蓋一個方法而繼承整個類,尤其是所需的執行個體(instantiation)很有限的時候。匿名類使得我們能在執行個體化的同時覆蓋方法。
如果我們只想在某個JButton對象中加入我們自己的可視邏輯,那麼我們可以在申明這個button對象的同時重寫這個方法:
JButton myButton = new JButton()
{
public void setVisible( boolean visible )
{
// Rolling our own visibility
}
};
這段代碼都做了什麼? 花括弧({})中間的代碼申明了setVisible()方法,並覆蓋了JButton類中的那個,但這僅限於myButton對象。我們沒有改變JButton類,也沒有申明一個新類,我們僅給了一個特殊的JButton對象它自己的可視邏輯。
在物件導向術語中,myButton是一個從JButton類繼承而來的無名,也就是匿名,類的對象。
這種建立匿名類並同時覆蓋方法的技術用在什麼時候?假設你在編寫一段Swing程式,在你向一個GUI物件(element)中添加一個 event listener(假設叫作ActionListener)之前,你已經編寫了一段這種機制的代碼。現在,我們假設我們有個龐大的類,裡面有很多按鈕,但是有一個按鈕時隱時現,你想知道為什麼會出這樣的異常情況,利用上面的代碼並在setVisible()方法上設定斷點。然後,當你運行你的程式時,你設定的斷點就會在恰當的地方暫停程式。檢查棧軌跡(stack trace),我們會發現沒有按所預期的那樣來調用setVisible()方法的原因並修複這個它。
匿名類在debug類似這種源碼不可得的類的時候很有用。即便在源碼可得的情況下,在大量使用的方法(如setVisible)上設定斷點,也是件很麻煩的事情,因為我們在每個實現了setVisible()方法的類的對象上都要轉入斷點。而匿名類可針對某個特定的對象進行“外科手術”式的 debug
內部類
簡單地說,“內部類”是在另一個類的內部聲明的類。從Java 1.1開始,你可在一個類中聲明另一個類,這與聲明欄位和方法非常相似。封裝了內部類聲明的類就稱為“外部類”。
實際上,Java語言規範還允許你做更多的事情,包括:
在另一個類或者一個介面中聲明一個類。
在另一個介面或者一個類中聲明一個介面。
在一個方法中聲明一個類。
類和介面聲明可嵌套任意深度。 內部類實現更最佳化的代碼
內部和匿名類是Java 1.1為我們提供的兩個出色的工具。它們提供了更好的封裝,結果就是使代碼更容易理解和維護,使相關的類都能存在於同一個原始碼檔案中(這要歸功於內部類),並能避免一個程式產生大量非常小的類(這要歸功於匿名類)。