匿名類是不能有名稱的類,所以沒辦法引用它們。必須在創(chuàng)建時,作為new語句的一部分來聲明它們。這就要采用另一種形式的new語句,如下所示: new <類或接口> <類的主體> 這種形式的new語句聲明一個新的匿名類,它對一個給定的類進(jìn)行擴(kuò)展,或者實現(xiàn)一個給定的接口。它還創(chuàng)建那個類的一個新實例,并把它作為語句的結(jié)果而返回。要擴(kuò)展的類和要實現(xiàn)的接口是new語句的操作數(shù),后跟匿名類的主體。如果匿名類對另一個類進(jìn)行擴(kuò)展,它的主體可以訪問類的成員、覆蓋它的方法等等,這和其他任何標(biāo)準(zhǔn)的類都是一樣的。如果匿名類實現(xiàn)了一個接口,它的主體必須實現(xiàn)接口的方法。
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也可以是一個類但是你外部調(diào)用的方法必須在你的這個類或接口中聲明外部不能調(diào)用匿名類內(nèi)部的方法
Java中內(nèi)部匿名類用的最多的地方也許就是在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);
}
}
內(nèi)部匿名類,就是建立一個內(nèi)部的類,但沒有給你命名,也就是沒有引用實例的變量。
new WindowAdapter() {
public void windowClosing(WindowEvent e) {
dispose();
System.exit(0);
}
}
new 是建立一個 WindowAdapter對象 ,后面一個 {} 表示這個括號中的操作作用于這個默認(rèn)的對名象,而上面的Java程序中后面是一個函數(shù)體。
這個用法的作用是:創(chuàng)建一個對象的實例,并且 override 它的一個函數(shù)。打開 WindowAdapter 的代碼可以發(fā)現(xiàn)。它是一個抽象類。它是對 WindowListener 接口的一個實現(xiàn)。Frame.addWindowListner(); 的參數(shù)是一個 WindowListner ,而實現(xiàn)上是傳一個從WindowAdapter 派生出的一個匿名類。
1.怎樣判斷一個匿名類的存在啊?看不見名字,感覺只是父類new出一個對象而已,沒有匿名類的名字。
先看段偽代碼
abstract class Father(){
....
}
public class Test{
Father f1 = new Father(){ .... } //這里就是有個匿名內(nèi)部類
}
一般來說,new 一個對象時小括號后應(yīng)該是分號,也就是new出對象該語句就結(jié)束了。
但是出現(xiàn)匿名內(nèi)部類就不一樣,小括號后跟的是大括號,大括號中是該new 出對象的具體的實現(xiàn)方法。
因為我們知道,一個抽象類是不能直接new 的,必須先有實現(xiàn)類了我們才能new出它的實現(xiàn)類。
上面的偽代碼就是表示new 的是Father的實現(xiàn)類,這個實現(xiàn)類是個匿名內(nèi)部類。
其實拆分上面的匿名內(nèi)部類可為
class SonOne extends Father{
...//這里的代碼和上面匿名內(nèi)部類,大括號中的代碼是一樣的
}
public class Test{
Father f1 = new SonOne() ;
}
2.匿名內(nèi)部類的注意事項
注意匿名類的聲明是在編譯時進(jìn)行的,實例化在運行時進(jìn)行。這意味著for循環(huán)中的一個new語句會創(chuàng)建相同匿名類的幾個實例,而不是創(chuàng)建幾個不同匿名類的一個實例。
在使用匿名內(nèi)部類時,要記住以下幾個原則:
?匿名內(nèi)部類不能有構(gòu)造方法。
?匿名內(nèi)部類不能定義任何靜態(tài)成員、方法和類。
?匿名內(nèi)部類不能是public,protected,private,static。
?只能創(chuàng)建匿名內(nèi)部類的一個實例。
?一個匿名內(nèi)部類一定是在new的后面,用其隱含實現(xiàn)一個接口或?qū)崿F(xiàn)一個類。
?因匿名內(nèi)部類為局部內(nèi)部類,所以局部內(nèi)部類的所有限制都對其生效。
?內(nèi)部類只能訪問外部類的靜態(tài)變量或靜態(tài)方法。
匿名類和內(nèi)部類中的中的this :
有時候,我們會用到一些內(nèi)部類和匿名類。當(dāng)在匿名類中用this時,這個this則指的是匿名類或內(nèi)部類本身。這時如果我們要使用外部類的方法和變量的話,則應(yīng)該加上外部類的類名
3.匿名內(nèi)部類的作用
Java的內(nèi)部類和C++中的嵌套類有本質(zhì)的不同:C++的嵌套類沒有指向包裝類的句柄。僅僅表達(dá)一個封裝的概念;但是Java的內(nèi)部類不同,它可以訪問包裝類的成員(這表示它擁有指向包裝類的句柄)。
匿名內(nèi)部類是內(nèi)部類的一種簡化寫法:return new Wrapper {
...
};
等價于:Wrapped extends Wrapper {
...
}
return new Wrapped();
難道匿名內(nèi)部類就只這一點作用嗎?
考慮一下這樣的case:
interface ICount {
int count();
}
class Parent {
int i = 0;
int count() {
return i++;
}
}
有一個類Child,它既想繼承Parent的count()方法,又想實現(xiàn)ICount接口中的count方法,這個時候怎么辦呢?內(nèi)部類就可以大顯身手了:
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(); // 創(chuàng)建一個對象
SwingUtilities.invokeLater(new Runnable() { // Anonymous Runnable class
// 一個匿名內(nèi)部類,他實現(xiàn)了一個線程
// 原本這個方法是傳一個Runnable類型參數(shù) // 這里可以通過這種匿名類的方式來實現(xiàn)
// object
public void run() { // Run method executed in thread
theApp.creatGUI(); // Call static GUI creator
}
});
}