在java提高篇-----詳解內(nèi)部類中對匿名內(nèi)部類做了一個簡單的介紹,但是內(nèi)部類還存在很多其他細節(jié)問題,所以就衍生出這篇博客。在這篇博客中你可以了解到匿名內(nèi)部類的使用、匿名內(nèi)部類要注意的事項、如何初始化匿名內(nèi)部類、匿名內(nèi)部類使用的形參為何要為final。
匿名內(nèi)部類由于沒有名字,所以它的創(chuàng)建方式有點兒奇怪。創(chuàng)建格式如下:
new 父類構(gòu)造器(參數(shù)列表)|實現(xiàn)接口() { //匿名內(nèi)部類的類體部分 } 在這里我們看到使用匿名內(nèi)部類我們必須要繼承一個父類或者實現(xiàn)一個接口,當然也僅能只繼承一個父類或者實現(xiàn)一個接口。同時它也是沒有class關(guān)鍵字,這是因為匿名內(nèi)部類是直接使用new來生成一個對象的引用。當然這個引用是隱式的。

public abstract class Bird { PRivate String name; public String getName() { return name; } public void setName(String name) { this.name = name; } public abstract int fly();}public class Test { public void test(Bird bird){ System.out.println(bird.getName() + "能夠飛 " + bird.fly() + "米"); } public static void main(String[] args) { Test test = new Test(); test.test(new Bird() { public int fly() { return 10000; } public String getName() { return "大雁"; } }); }}------------------Output:大雁能夠飛 10000米
在Test類中,test()方法接受一個Bird類型的參數(shù),同時我們知道一個抽象類是沒有辦法直接new的,我們必須要先有實現(xiàn)類才能new出來它的實現(xiàn)類實例。所以在mian方法中直接使用匿名內(nèi)部類來創(chuàng)建一個Bird實例。
由于匿名內(nèi)部類不能是抽象類,所以它必須要實現(xiàn)它的抽象父類或者接口里面所有的抽象方法。
對于這段匿名內(nèi)部類代碼其實是可以拆分為如下形式:

public class WildGoose extends Bird{ public int fly() { return 10000; } public String getName() { return "大雁"; }}WildGoose wildGoose = new WildGoose();test.test(wildGoose);
在這里系統(tǒng)會創(chuàng)建一個繼承自Bird類的匿名類的對象,該對象轉(zhuǎn)型為對Bird類型的引用。
對于匿名內(nèi)部類的使用它是存在一個缺陷的,就是它僅能被使用一次,創(chuàng)建匿名內(nèi)部類時它會立即創(chuàng)建一個該類的實例,該類的定義會立即消失,所以匿名內(nèi)部類是不能夠被重復使用。對于上面的實例,如果我們需要對test()方法里面內(nèi)部類進行多次使用,建議重新定義類,而不是使用匿名內(nèi)部類。
二、注意事項
在使用匿名內(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)的接口的所有抽象方法。
三、使用的形參為何要為final
參考文件:http://android.blog.51cto.com/268543/384844
我們給匿名內(nèi)部類傳遞參數(shù)的時候,若該形參在內(nèi)部類中需要被使用,那么該形參必須要為final。也就是說:當所在的方法的形參需要被內(nèi)部類里面使用時,該形參必須為final。
為什么必須要為final呢?
首先我們知道在內(nèi)部類編譯成功后,它會產(chǎn)生一個class文件,該class文件與外部類并不是同一class文件,僅僅只保留對外部類的引用。當外部類傳入的參數(shù)需要被內(nèi)部類調(diào)用時,從java程序的角度來看是直接被調(diào)用:

public class OuterClass { public void display(final String name,String age){ class InnerClass{ void display(){ System.out.println(name); } } }}
從上面代碼中看好像name參數(shù)應該是被內(nèi)部類直接調(diào)用?其實不然,在java編譯之后實際的操作如下:

public class OuterClass$InnerClass { public InnerClass(String name,String age){ this.InnerClass$name = name; this.InnerClass$age = age; } public void display(){ System.out.println(this.InnerClass$name + "----" + this.InnerClass$age ); }}
所以從上面代碼來看,內(nèi)部類并不是直接調(diào)用方法傳遞的參數(shù),而是利用自身的構(gòu)造器對傳入的參數(shù)進行備份,自己內(nèi)部方法調(diào)用的實際上時自己的屬性而不是外部方法傳遞進來的參數(shù)。
直到這里還沒有解釋為什么是final?在內(nèi)部類中的屬性和外部方法的參數(shù)兩者從外表上看是同一個東西,但實際上卻不是,所以他們兩者是可以任意變化的,也就是說在內(nèi)部類中我對屬性的改變并不會影響到外部的形參,而然這從程序員的角度來看這是不可行的,畢竟站在程序的角度來看這兩個根本就是同一個,如果內(nèi)部類該變了,而外部方法的形參卻沒有改變這是難以理解和不可接受的,所以為了保持參數(shù)的一致性,就規(guī)定使用final來避免形參的不改變。
簡單理解就是,拷貝引用,為了避免引用值發(fā)生改變,例如被外部類的方法修改等,而導致內(nèi)部類得到的值不一致,于是用final來讓該引用不可改變。
故如果定義了一個匿名內(nèi)部類,并且希望它使用一個其外部定義的參數(shù),那么編譯器會要求該參數(shù)引用是final的。
四、匿名內(nèi)部類初始化
我們一般都是利用構(gòu)造器來完成某個實例的初始化工作的,但是匿名內(nèi)部類是沒有構(gòu)造器的!那怎么來初始化匿名內(nèi)部類呢?使用構(gòu)造代碼塊!利用構(gòu)造代碼塊能夠達到為匿名內(nèi)部類創(chuàng)建一個構(gòu)造器的效果。

public class OutClass { public InnerClass getInnerClass(final int age,final String name){ return new InnerClass() { int age_ ; String name_; //構(gòu)造代碼塊完成初始化工作 { if(0 < age && age < 200){ age_ = age; name_ = name; } } public String getName() { return name_; } public int getAge() { return age_; } }; } public static void main(String[] args) { OutClass out = new OutClass(); InnerClass inner_1 = out.getInnerClass(201, "chenssy"); System.out.println(inner_1.getName()); InnerClass inner_2 = out.getInnerClass(23, "chenssy"); System.out.println(inner_2.getName()); }}
鞏固基礎(chǔ),提高技術(shù),不懼困難,攀登高峰!!!!!!
PS:如果你覺得文章對你有所幫助,別忘了推薦或者分享,因為有你的支持,才是我續(xù)寫下篇的動力和源泉!
新聞熱點
疑難解答