1. 程式人生 > >阿里P7講解Java進階之詳解匿名內部類

阿里P7講解Java進階之詳解匿名內部類

在java提高篇—–詳解內部類中對匿名內部類做了一個簡單的介紹,但是內部類還存在很多其他細節問題,所以就衍生出這篇部落格。在這篇部落格中你可以瞭解到匿名內部類的使用、匿名內部類要注意的事項、如何初始化匿名內部類、匿名內部類使用的形參為何要為final。

一、使用匿名內部類內部類

匿名內部類由於沒有名字,所以它的建立方式有點兒奇怪。建立格式如下:

new 父類構造器(引數列表)|實現介面() 
 { 
 //匿名內部類的類體部分 
 }

在這裡我們看到使用匿名內部類我們必須要繼承一個父類或者實現一個介面,當然也僅能只繼承一個父類或者實現一個介面。同時它也是沒有class關鍵字,這是因為匿名內部類是直接使用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型別的引數,同時我們知道一個抽象類是沒有辦法直接new的,我們必須要先有實現類才能new出來它的實現類例項。所以在mian方法中直接使用匿名內部類來建立一個Bird例項。

由於匿名內部類不能是抽象類,所以它必須要實現它的抽象父類或者接口裡面所有的抽象方法。

對於這段匿名內部類程式碼其實是可以拆分為如下形式:

public class WildGoose extends Bird{
 public int fly() {
 return 10000;
 }
 public String getName() {
 return "大雁";
 }
}
WildGoose wildGoose = new WildGoose();
test.test(wildGoose);

在這裡系統會建立一個繼承自Bird類的匿名類的物件,該物件轉型為對Bird型別的引用。

對於匿名內部類的使用它是存在一個缺陷的,就是它僅能被使用一次,建立匿名內部類時它會立即建立一個該類的例項,該類的定義會立即消失,所以匿名內部類是不能夠被重複使用。對於上面的例項,如果我們需要對test()方法裡面內部類進行多次使用,建議重新定義類,而不是使用匿名內部類。

二、注意事項

在使用匿名內部類的過程中,我們需要注意如下幾點:

1、使用匿名內部類時,我們必須是繼承一個類或者實現一個介面,但是兩者不可兼得,同時也只能繼承一個類或者實現一個介面。
2、匿名內部類中是不能定義建構函式的。
3、匿名內部類中不能存在任何的靜態成員變數和靜態方法。
4、匿名內部類為區域性內部類,所以區域性內部類的所有限制同樣對匿名內部類生效。
5、匿名內部類不能是抽象的,它必須要實現繼承的類或者實現的介面的所有抽象方法。

三、使用的形參為何要為final

我們給匿名內部類傳遞引數的時候,若該形參在內部類中需要被使用,那麼該形參必須要為final。也就是說:當所在的方法的形參需要被內部類裡面使用時,該形參必須為final。

為什麼必須要為final呢?

首先我們知道在內部類編譯成功後,它會產生一個class檔案,該class檔案與外部類並不是同一class檔案,僅僅只保留對外部類的引用。當外部類傳入的引數需要被內部類呼叫時,從java程式的角度來看是直接被呼叫:

public class OuterClass {
 public void display(final String name,String age){
 class InnerClass{
 void display(){
 System.out.println(name);
 }
 }
 }
}

從上面程式碼中看好像name引數應該是被內部類直接呼叫?其實不然,在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 );
 }
}

所以從上面程式碼來看,內部類並不是直接呼叫方法傳遞的引數,而是利用自身的構造器對傳入的引數進行備份,自己內部方法呼叫的實際上時自己的屬性而不是外部方法傳遞進來的引數。

直到這裡還沒有解釋為什麼是final?在內部類中的屬性和外部方法的引數兩者從外表上看是同一個東西,但實際上卻不是,所以他們兩者是可以任意變化的,也就是說在內部類中我對屬性的改變並不會影響到外部的形參,而然這從程式設計師的角度來看這是不可行的,畢竟站在程式的角度來看這兩個根本就是同一個,如果內部類該變了,而外部方法的形參卻沒有改變這是難以理解和不可接受的,所以為了保持引數的一致性,就規定使用final來避免形參的不改變。

簡單理解就是,拷貝引用,為了避免引用值發生改變,例如被外部類的方法修改等,而導致內部類得到的值不一致,於是用final來讓該引用不可改變。

故如果定義了一個匿名內部類,並且希望它使用一個其外部定義的引數,那麼編譯器會要求該引數引用是final的。

四、匿名內部類初始化

我們一般都是利用構造器來完成某個例項的初始化工作的,但是匿名內部類是沒有構造器的!那怎麼來初始化匿名內部類呢?使用構造程式碼塊!利用構造程式碼塊能夠達到為匿名內部類建立一個構造器的效果。

public class OutClass {
 public InnerClass getInnerClass(final int age,final String name){
 return new InnerClass() {
 int age_ ;
 String name_;
 //構造程式碼塊完成初始化工作
 {
 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());
 }
}