java匿名內(nèi)部類實(shí)例簡(jiǎn)析
更新時(shí)間:2012年11月28日 15:05:46 作者:
匿名類是不能有名稱的類,所以沒辦法引用它們,必須在創(chuàng)建時(shí),作為new語句的一部分來聲明它們,需要了解更多的可以參考本文
匿名類是不能有名稱的類,所以沒辦法引用它們。必須在創(chuàng)建時(shí),作為new語句的一部分來聲明它們。這就要采用另一種形式的new語句,如下所示: new <類或接口> <類的主體> 這種形式的new語句聲明一個(gè)新的匿名類,它對(duì)一個(gè)給定的類進(jìn)行擴(kuò)展,或者實(shí)現(xiàn)一個(gè)給定的接口。它還創(chuàng)建那個(gè)類的一個(gè)新實(shí)例,并把它作為語句的結(jié)果而返回。要擴(kuò)展的類和要實(shí)現(xiàn)的接口是new語句的操作數(shù),后跟匿名類的主體。如果匿名類對(duì)另一個(gè)類進(jìn)行擴(kuò)展,它的主體可以訪問類的成員、覆蓋它的方法等等,這和其他任何標(biāo)準(zhǔn)的類都是一樣的。如果匿名類實(shí)現(xiàn)了一個(gè)接口,它的主體必須實(shí)現(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也可以是一個(gè)類但是你外部調(diào)用的方法必須在你的這個(gè)類或接口中聲明外部不能調(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)部匿名類,就是建立一個(gè)內(nèi)部的類,但沒有給你命名,也就是沒有引用實(shí)例的變量。
new WindowAdapter() {
public void windowClosing(WindowEvent e) {
dispose();
System.exit(0);
}
}
new 是建立一個(gè) WindowAdapter對(duì)象 ,后面一個(gè) {} 表示這個(gè)括號(hào)中的操作作用于這個(gè)默認(rèn)的對(duì)名象,而上面的Java程序中后面是一個(gè)函數(shù)體。
這個(gè)用法的作用是:創(chuàng)建一個(gè)對(duì)象的實(shí)例,并且 override 它的一個(gè)函數(shù)。打開 WindowAdapter 的代碼可以發(fā)現(xiàn)。它是一個(gè)抽象類。它是對(duì) WindowListener 接口的一個(gè)實(shí)現(xiàn)。Frame.addWindowListner(); 的參數(shù)是一個(gè) WindowListner ,而實(shí)現(xiàn)上是傳一個(gè)從WindowAdapter 派生出的一個(gè)匿名類。
1.怎樣判斷一個(gè)匿名類的存在啊?看不見名字,感覺只是父類new出一個(gè)對(duì)象而已,沒有匿名類的名字。
先看段偽代碼
abstract class Father(){
....
}
public class Test{
Father f1 = new Father(){ .... } //這里就是有個(gè)匿名內(nèi)部類
}
一般來說,new 一個(gè)對(duì)象時(shí)小括號(hào)后應(yīng)該是分號(hào),也就是new出對(duì)象該語句就結(jié)束了。
但是出現(xiàn)匿名內(nèi)部類就不一樣,小括號(hào)后跟的是大括號(hào),大括號(hào)中是該new 出對(duì)象的具體的實(shí)現(xiàn)方法。
因?yàn)槲覀冎溃粋€(gè)抽象類是不能直接new 的,必須先有實(shí)現(xiàn)類了我們才能new出它的實(shí)現(xiàn)類。
上面的偽代碼就是表示new 的是Father的實(shí)現(xiàn)類,這個(gè)實(shí)現(xiàn)類是個(gè)匿名內(nèi)部類。
其實(shí)拆分上面的匿名內(nèi)部類可為
class SonOne extends Father{
...//這里的代碼和上面匿名內(nèi)部類,大括號(hào)中的代碼是一樣的
}
public class Test{
Father f1 = new SonOne() ;
}
2.匿名內(nèi)部類的注意事項(xiàng)
注意匿名類的聲明是在編譯時(shí)進(jìn)行的,實(shí)例化在運(yùn)行時(shí)進(jìn)行。這意味著for循環(huán)中的一個(gè)new語句會(huì)創(chuàng)建相同匿名類的幾個(gè)實(shí)例,而不是創(chuàng)建幾個(gè)不同匿名類的一個(gè)實(shí)例。
在使用匿名內(nèi)部類時(shí),要記住以下幾個(gè)原則:
·匿名內(nèi)部類不能有構(gòu)造方法。
·匿名內(nèi)部類不能定義任何靜態(tài)成員、方法和類。
·匿名內(nèi)部類不能是public,protected,private,static。
·只能創(chuàng)建匿名內(nèi)部類的一個(gè)實(shí)例。
·一個(gè)匿名內(nèi)部類一定是在new的后面,用其隱含實(shí)現(xiàn)一個(gè)接口或?qū)崿F(xiàn)一個(gè)類。
·因匿名內(nèi)部類為局部?jī)?nèi)部類,所以局部?jī)?nèi)部類的所有限制都對(duì)其生效。
·內(nèi)部類只能訪問外部類的靜態(tài)變量或靜態(tài)方法。
匿名類和內(nèi)部類中的中的this :
有時(shí)候,我們會(huì)用到一些內(nèi)部類和匿名類。當(dāng)在匿名類中用this時(shí),這個(gè)this則指的是匿名類或內(nèi)部類本身。這時(shí)如果我們要使用外部類的方法和變量的話,則應(yīng)該加上外部類的類名
3.匿名內(nèi)部類的作用
Java的內(nèi)部類和C++中的嵌套類有本質(zhì)的不同:C++的嵌套類沒有指向包裝類的句柄。僅僅表達(dá)一個(gè)封裝的概念;但是Java的內(nèi)部類不同,它可以訪問包裝類的成員(這表示它擁有指向包裝類的句柄)。
匿名內(nèi)部類是內(nèi)部類的一種簡(jiǎn)化寫法:return new Wrapper {
...
};
等價(jià)于:Wrapped extends Wrapper {
...
}
return new Wrapped();
難道匿名內(nèi)部類就只這一點(diǎn)作用嗎?
考慮一下這樣的case:
interface ICount {
int count();
}
class Parent {
int i = 0;
int count() {
return i++;
}
}
有一個(gè)類Child,它既想繼承Parent的count()方法,又想實(shí)現(xiàn)ICount接口中的count方法,這個(gè)時(shí)候怎么辦呢??jī)?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)建一個(gè)對(duì)象
SwingUtilities.invokeLater(new Runnable() { // Anonymous Runnable class
// 一個(gè)匿名內(nèi)部類,他實(shí)現(xiàn)了一個(gè)線程
// 原本這個(gè)方法是傳一個(gè)Runnable類型參數(shù) // 這里可以通過這種匿名類的方式來實(shí)現(xiàn)
// object
public void run() { // Run method executed in thread
theApp.creatGUI(); // Call static GUI creator
}
});
}
java 代碼
復(fù)制代碼 代碼如下:
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也可以是一個(gè)類但是你外部調(diào)用的方法必須在你的這個(gè)類或接口中聲明外部不能調(diào)用匿名類內(nèi)部的方法
Java中內(nèi)部匿名類用的最多的地方也許就是在Frame中加入Listner了吧。
如下:
java 代碼
復(fù)制代碼 代碼如下:
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)部匿名類,就是建立一個(gè)內(nèi)部的類,但沒有給你命名,也就是沒有引用實(shí)例的變量。
復(fù)制代碼 代碼如下:
new WindowAdapter() {
public void windowClosing(WindowEvent e) {
dispose();
System.exit(0);
}
}
new 是建立一個(gè) WindowAdapter對(duì)象 ,后面一個(gè) {} 表示這個(gè)括號(hào)中的操作作用于這個(gè)默認(rèn)的對(duì)名象,而上面的Java程序中后面是一個(gè)函數(shù)體。
這個(gè)用法的作用是:創(chuàng)建一個(gè)對(duì)象的實(shí)例,并且 override 它的一個(gè)函數(shù)。打開 WindowAdapter 的代碼可以發(fā)現(xiàn)。它是一個(gè)抽象類。它是對(duì) WindowListener 接口的一個(gè)實(shí)現(xiàn)。Frame.addWindowListner(); 的參數(shù)是一個(gè) WindowListner ,而實(shí)現(xiàn)上是傳一個(gè)從WindowAdapter 派生出的一個(gè)匿名類。
1.怎樣判斷一個(gè)匿名類的存在啊?看不見名字,感覺只是父類new出一個(gè)對(duì)象而已,沒有匿名類的名字。
先看段偽代碼
復(fù)制代碼 代碼如下:
abstract class Father(){
....
}
public class Test{
Father f1 = new Father(){ .... } //這里就是有個(gè)匿名內(nèi)部類
}
一般來說,new 一個(gè)對(duì)象時(shí)小括號(hào)后應(yīng)該是分號(hào),也就是new出對(duì)象該語句就結(jié)束了。
但是出現(xiàn)匿名內(nèi)部類就不一樣,小括號(hào)后跟的是大括號(hào),大括號(hào)中是該new 出對(duì)象的具體的實(shí)現(xiàn)方法。
因?yàn)槲覀冎溃粋€(gè)抽象類是不能直接new 的,必須先有實(shí)現(xiàn)類了我們才能new出它的實(shí)現(xiàn)類。
上面的偽代碼就是表示new 的是Father的實(shí)現(xiàn)類,這個(gè)實(shí)現(xiàn)類是個(gè)匿名內(nèi)部類。
其實(shí)拆分上面的匿名內(nèi)部類可為
復(fù)制代碼 代碼如下:
class SonOne extends Father{
...//這里的代碼和上面匿名內(nèi)部類,大括號(hào)中的代碼是一樣的
}
public class Test{
Father f1 = new SonOne() ;
}
2.匿名內(nèi)部類的注意事項(xiàng)
注意匿名類的聲明是在編譯時(shí)進(jìn)行的,實(shí)例化在運(yùn)行時(shí)進(jìn)行。這意味著for循環(huán)中的一個(gè)new語句會(huì)創(chuàng)建相同匿名類的幾個(gè)實(shí)例,而不是創(chuàng)建幾個(gè)不同匿名類的一個(gè)實(shí)例。
在使用匿名內(nèi)部類時(shí),要記住以下幾個(gè)原則:
·匿名內(nèi)部類不能有構(gòu)造方法。
·匿名內(nèi)部類不能定義任何靜態(tài)成員、方法和類。
·匿名內(nèi)部類不能是public,protected,private,static。
·只能創(chuàng)建匿名內(nèi)部類的一個(gè)實(shí)例。
·一個(gè)匿名內(nèi)部類一定是在new的后面,用其隱含實(shí)現(xiàn)一個(gè)接口或?qū)崿F(xiàn)一個(gè)類。
·因匿名內(nèi)部類為局部?jī)?nèi)部類,所以局部?jī)?nèi)部類的所有限制都對(duì)其生效。
·內(nèi)部類只能訪問外部類的靜態(tài)變量或靜態(tài)方法。
匿名類和內(nèi)部類中的中的this :
有時(shí)候,我們會(huì)用到一些內(nèi)部類和匿名類。當(dāng)在匿名類中用this時(shí),這個(gè)this則指的是匿名類或內(nèi)部類本身。這時(shí)如果我們要使用外部類的方法和變量的話,則應(yīng)該加上外部類的類名
3.匿名內(nèi)部類的作用
Java的內(nèi)部類和C++中的嵌套類有本質(zhì)的不同:C++的嵌套類沒有指向包裝類的句柄。僅僅表達(dá)一個(gè)封裝的概念;但是Java的內(nèi)部類不同,它可以訪問包裝類的成員(這表示它擁有指向包裝類的句柄)。
匿名內(nèi)部類是內(nèi)部類的一種簡(jiǎn)化寫法:return new Wrapper {
...
};
等價(jià)于:Wrapped extends Wrapper {
...
}
return new Wrapped();
難道匿名內(nèi)部類就只這一點(diǎn)作用嗎?
考慮一下這樣的case:
復(fù)制代碼 代碼如下:
interface ICount {
int count();
}
class Parent {
int i = 0;
int count() {
return i++;
}
}
有一個(gè)類Child,它既想繼承Parent的count()方法,又想實(shí)現(xiàn)ICount接口中的count方法,這個(gè)時(shí)候怎么辦呢??jī)?nèi)部類就可以大顯身手了:
復(fù)制代碼 代碼如下:
class Child extends Parent {
ICount getCount() {
return new ICount {
int i = 0;
int count() {
return (i *= 2);
}
}
}
}
看這段代碼
復(fù)制代碼 代碼如下:
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)建一個(gè)對(duì)象
SwingUtilities.invokeLater(new Runnable() { // Anonymous Runnable class
// 一個(gè)匿名內(nèi)部類,他實(shí)現(xiàn)了一個(gè)線程
// 原本這個(gè)方法是傳一個(gè)Runnable類型參數(shù) // 這里可以通過這種匿名類的方式來實(shí)現(xiàn)
// object
public void run() { // Run method executed in thread
theApp.creatGUI(); // Call static GUI creator
}
});
}
相關(guān)文章
Java使用DOM4j實(shí)現(xiàn)讀寫XML文件的屬性和元素
這篇文章主要為大家詳細(xì)介紹了Java使用DOM4j實(shí)現(xiàn)讀寫XML文件的屬性和元素,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2018-12-12servlet實(shí)現(xiàn)文件上傳、預(yù)覽、下載、刪除功能
這篇文章主要為大家詳細(xì)介紹了servlet實(shí)現(xiàn)文件上傳、預(yù)覽、下載、刪除功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-09-09springboot通過jar包啟動(dòng)中文日志亂碼問題及解決
這篇文章主要介紹了springboot通過jar包啟動(dòng)中文日志亂碼問題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-06-06IntelliJ Plugin 開發(fā)之添加第三方j(luò)ar的示例代碼
這篇文章主要介紹了IntelliJ Plugin 開發(fā)之添加第三方j(luò)ar的示例代碼,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-09-09Java實(shí)現(xiàn)簡(jiǎn)單樹結(jié)構(gòu)
這篇文章主要為大家詳細(xì)介紹了Java實(shí)現(xiàn)簡(jiǎn)單樹結(jié)構(gòu)的相關(guān)資料,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-01-01