在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()); } }
以上就是 java提高篇(九)-----详解匿名内部类的内容,更多相关内容请关注PHP中文网(www.php.cn)!