匿名類別是不能有名稱的類,所以沒辦法引用它們。必須在創建時,作為new語句的一部分來聲明它們。這就要採用另一種形式的new語句,如下所示: new <類或介面> <類的主體> 這種形式的new語句聲明一個新的匿名類,它對一個給定的類進行擴展,或實作一個給定的介面。它也創建那個類別的一個新實例,並把它當作語句的結果而傳回。要擴充的類別和要實作的介面是new語句的操作數,後面跟著匿名類別的主體。如果匿名類別對另一個類別進行擴展,它的主體可以存取類別的成員、覆蓋它的方法等等,這和其他任何標準的類別都是一樣的。如果匿名類別實作了一個接口,它的主體必須實作接口的方法。
java 程式碼
複製代碼代碼如下:
interface pr
{
void print1();
}
public class noNameClass
{
public pr dest()
{
return new pr(){
public void print1()
{
System.out.println("Hello world!!");
}
};
}
public static void main(String args[])
{
noNameClass c = new noNameClass();
pr hw=c.dest();
hw.print1();
}
}
pr也可以是一個類別但是你外部呼叫的方法必須在你的這個類別或介面中宣告外部不能呼叫匿名類別內部的方法
Java中內部匿名類別用的最多的地方也許就是在Frame中加入Listner了吧。
如下:
java 程式碼
複製代碼代碼如下:
import java.awt.*;
import java.awt.event.*;
public class QFrame extends Frame {
public QFrame() {
this.setTitle(/"my application/");
addWindowListener(new WindowAdapter() {
public void windowClosing(WindowEvent e) {
dispose();
System.exit(0);
}
});
this.setBounds(10,10,200,200);
}
}
內部匿名類,就是建立一個內部的類,但沒有為你命名,也就是沒有引用實例的變數。
複製代碼代碼如下:
new WindowAdapter() {
public void windowClosing(WindowEvent e) {
dispose();
System.exit(0);
}
}
new 是建立一個WindowAdapter對象,後面一個{} 表示這個括號中的運算作用於這個預設的對名象,而上面的Java程式中後面是一個函數體。
這個用法的作用是:建立一個物件的實例,而override 它的一個函數。開啟WindowAdapter 的程式碼可以發現。它是一個抽象類別。它是對WindowListener 介面的一個實作。 Frame.addWindowListner(); 的參數是一個WindowListner ,而實作上是傳一個從WindowAdapter 衍生出的一個匿名類別。
1.怎樣判斷一個匿名類別的存在啊?看不見名字,感覺只是父類別new出一個物件而已,沒有匿名類別的名字。
先看段偽代碼
複製代碼代碼如下:
abstract class Father(){
....
}
public class Test{
Father f1 = new Father(){ .... } //這裡就是有個匿名內部類
}
一般來說,new 一個物件時小括號後面應該是分號,也就是new出物件該語句就結束了。
但出現匿名內部類別就不一樣,小括號後面接的是大括號,大括號中是該new 出物件的具體的實作方法。
因為我們知道,一個抽象類別是不能直接new 的,必須先有實作類別了我們才能new出它的實作類別。
上面的偽代碼就是表示new 的是Father的實作類,這個實作類別是個匿名內部類別。
其實拆分上面的匿名內部類別可為
複製代碼代碼如下:
class SonOne extends Father{
...//這裡的程式碼和上面匿名內部類,大括號中的程式碼是一樣的
}
public class Test{
Father f1 = new SonOne() ;
}
2.匿名內部類別的注意事項注意匿名類別的聲明是在編譯時進行的,實例化在執行時進行。這表示for迴圈中的一個new語句會建立相同匿名類別的幾個實例,而不是建立幾個不同匿名類別的一個實例。
在使用匿名內部類別時,請記住以下幾個原則:
・匿名內部類別不能有構造方法。
・匿名內部類別不能定義任何靜態成員、方法和類別。
・匿名內部類別不能是public,protected,private,static。
・只能建立匿名內部類別的一個實例。
・一個匿名內部類別一定是在new的後面,用其隱含實作一個介面或實作一個類別。
・因匿名內部類為局部內部類,所以局部內部類別的所有限制都對其生效。
・內部類別只能存取外部類別的靜態變數或靜態方法。
匿名類別和內部類別中的中的this :
有時候,我們會用到一些內部類別和匿名類別。在匿名類別中用this時,這個this則指的是匿名類別或內部類別本身。這時如果我們要使用外部類別的方法和變數的話,則應該加上外部類別的類別名
3.匿名內部類別的作用
Java的內部類別和C++中的巢狀類別有本質的不同:C++的巢狀類別沒有指向包裝類別的句柄。僅僅表達一個封裝的概念;但是Java的內部類別不同,它可以存取包裝類別的成員(這表示它擁有指向包裝類別的句柄)。
匿名內部類別是內部類別的一種簡化寫法:return new Wrapper {
……
};
等價於:Wrapped extends Wrapper {
……
}
return new Wrapped();
難道匿名內部類別就只這一點作用嗎?
考慮一下這樣的case:
複製代碼代碼如下:
interface ICount {
int count();
}
class Parent {
int i = 0;
int count() {
return i++;
}
}
有一個類別Child,它既想繼承Parent的count()方法,又想實作ICount介面中的count方法,這時候該怎麼辦呢?內部類別就可以大顯身手了:
複製代碼代碼如下:
class Child extends Parent {
ICount getCount() {
return new ICount {
int i = 0;
int count() {
return (i *= 2);
}
}
}
}
看這段程式碼
複製代碼代碼如下:
public static void main(String[] args) {
theApp = new Analyzer();
SwingUtilities.invokeLater(new Runnable() { // Anonymous Runnable class
// object
public void run() { // Run method executed in thread
theApp.creatGUI(); // Call static GUI creator
}
});
}
public static void main(String[] args) {
theApp = new Analyzer(); // 建立一個對象
SwingUtilities.invokeLater(new Runnable() { // Anonymous Runnable class
// 一個匿名內部類,他實作了一個線程
// 原本這個方法是傳一個Runnable型別參數// 這裡可以透過這種匿名類別的方式來實現
// object
public void run() { // Run method executed in thread
theApp.creatGUI(); // Call static GUI creator
}
});
}