鍍金池/ 教程/ Java/ Java 提高篇(十)—–詳解匿名內(nèi)部類
Java 集合細(xì)節(jié)(四):保持 compareTo 和 equals 同步
Iterator
使用序列化實(shí)現(xiàn)對(duì)象的拷貝
fail-fast 機(jī)制
關(guān)鍵字 final
Vector
HashTable
Java 集合細(xì)節(jié)(一):請(qǐng)為集合指定初始容量
強(qiáng)制類型轉(zhuǎn)換
數(shù)組之一:認(rèn)識(shí) JAVA 數(shù)組
Java 集合細(xì)節(jié)(三):subList 的缺陷
hashCode
ArrayList
數(shù)組之二
List 總結(jié)
LinkedList
Java 提高篇(九)—–實(shí)現(xiàn)多重繼承
Java 的四舍五入
關(guān)鍵字 static
理解 Java 的三大特性之多態(tài)
抽象類與接口
集合大家族
異常(二)
Java 集合細(xì)節(jié)(二):asList 的缺陷
Map 總結(jié)
TreeSet
equals() 方法總結(jié)
Java 提高篇(十)—–詳解匿名內(nèi)部類
HashMap
Stack
詳解內(nèi)部類
TreeMap
異常(一)
詳解 Java 定時(shí)任務(wù)
HashSet
字符串
理解 Java 的三大特性之繼承
理解 Java 的三大特性之封裝
代碼塊

Java 提高篇(十)—–詳解匿名內(nèi)部類

在 Java 提高篇—–詳解內(nèi)部類中對(duì)匿名內(nèi)部類做了一個(gè)簡(jiǎn)單的介紹,但是內(nèi)部類還存在很多其他細(xì)節(jié)問題,所以就衍生出這篇博客。在這篇博客中你可以了解到匿名內(nèi)部類的使用、匿名內(nèi)部類要注意的事項(xiàng)、如何初始化匿名內(nèi)部類、匿名內(nèi)部類使用的形參為何要為 final。

一、使用匿名內(nèi)部類內(nèi)部類

匿名內(nèi)部類由于沒有名字,所以它的創(chuàng)建方式有點(diǎn)兒奇怪。創(chuàng)建格式如下:


    new 父類構(gòu)造器(參數(shù)列表)|實(shí)現(xiàn)接口()  
        {  
         //匿名內(nèi)部類的類體部分  
        }

在這里我們看到使用匿名內(nèi)部類我們必須要繼承一個(gè)父類或者實(shí)現(xiàn)一個(gè)接口,當(dāng)然也僅能只繼承一個(gè)父類或者實(shí)現(xiàn)一個(gè)接口。同時(shí)它也是沒有 class 關(guān)鍵字,這是因?yàn)槟涿麅?nèi)部類是直接使用 new 來生成一個(gè)對(duì)象的引用。當(dāng)然這個(gè)引用是隱式的。


    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() 方法接受一個(gè) Bird 類型的參數(shù),同時(shí)我們知道一個(gè)抽象類是沒有辦法直接 new 的,我們必須要先有實(shí)現(xiàn)類才能 new 出來它的實(shí)現(xiàn)類實(shí)例。所以在 main 方法中直接使用匿名內(nèi)部類來創(chuàng)建一個(gè) Bird 實(shí)例。

由于匿名內(nèi)部類不能是抽象類,所以它必須要實(shí)現(xiàn)它的抽象父類或者接口里面所有的抽象方法。

對(duì)于這段匿名內(nèi)部類代碼其實(shí)是可以拆分為如下形式:


    public class WildGoose extends Bird{
        public int fly() {
            return 10000;
        }

        public String getName() {
            return "大雁";
        }
    }

    WildGoose wildGoose = new WildGoose();
    test.test(wildGoose);

在這里系統(tǒng)會(huì)創(chuàng)建一個(gè)繼承自 Bird 類的匿名類的對(duì)象,該對(duì)象轉(zhuǎn)型為對(duì) Bird 類型的引用。

對(duì)于匿名內(nèi)部類的使用它是存在一個(gè)缺陷的,就是它僅能被使用一次,創(chuàng)建匿名內(nèi)部類時(shí)它會(huì)立即創(chuàng)建一個(gè)該類的實(shí)例,該類的定義會(huì)立即消失,所以匿名內(nèi)部類是不能夠被重復(fù)使用。對(duì)于上面的實(shí)例,如果我們需要對(duì) test() 方法里面內(nèi)部類進(jìn)行多次使用,建議重新定義類,而不是使用匿名內(nèi)部類。

二、注意事項(xiàng)

在使用匿名內(nèi)部類的過程中,我們需要注意如下幾點(diǎn):

1、使用匿名內(nèi)部類時(shí),我們必須是繼承一個(gè)類或者實(shí)現(xiàn)一個(gè)接口,但是兩者不可兼得,同時(shí)也只能繼承一個(gè)類或者實(shí)現(xiàn)一個(gè)接口。

2、匿名內(nèi)部類中是不能定義構(gòu)造函數(shù)的。

3、匿名內(nèi)部類中不能存在任何的靜態(tài)成員變量和靜態(tài)方法。

4、匿名內(nèi)部類為局部?jī)?nèi)部類,所以局部?jī)?nèi)部類的所有限制同樣對(duì)匿名內(nèi)部類生效。

5、匿名內(nèi)部類不能是抽象的,它必須要實(shí)現(xiàn)繼承的類或者實(shí)現(xiàn)的接口的所有抽象方法。

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

參考文件:http://android.blog.51cto.com/268543/384844

我們給匿名內(nèi)部類傳遞參數(shù)的時(shí)候,若該形參在內(nèi)部類中需要被使用,那么該形參必須要為 final。也就是說:當(dāng)所在的方法的形參需要被內(nèi)部類里面使用時(shí),該形參必須為 final。

為什么必須要為 final 呢?

首先我們知道在內(nèi)部類編譯成功后,它會(huì)產(chǎn)生一個(gè) class 文件,該 class 文件與外部類并不是同一 class 文件,僅僅只保留對(duì)外部類的引用。當(dāng)外部類傳入的參數(shù)需要被內(nèi)部類調(diào)用時(shí),從 java 程序的角度來看是直接被調(diào)用:


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

從上面代碼中看好像 name 參數(shù)應(yīng)該是被內(nèi)部類直接調(diào)用?其實(shí)不然,在 java 編譯之后實(shí)際的操作如下:


    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)造器對(duì)傳入的參數(shù)進(jìn)行備份,自己內(nèi)部方法調(diào)用的實(shí)際上時(shí)自己的屬性而不是外部方法傳遞進(jìn)來的參數(shù)。

直到這里還沒有解釋為什么是 final?在內(nèi)部類中的屬性和外部方法的參數(shù)兩者從外表上看是同一個(gè)東西,但實(shí)際上卻不是,所以他們兩者是可以任意變化的,也就是說在內(nèi)部類中我對(duì)屬性的改變并不會(huì)影響到外部的形參,而然這從程序員的角度來看這是不可行的,畢竟站在程序的角度來看這兩個(gè)根本就是同一個(gè),如果內(nèi)部類該變了,而外部方法的形參卻沒有改變這是難以理解和不可接受的,所以為了保持參數(shù)的一致性,就規(guī)定使用 final 來避免形參的不改變。

簡(jiǎn)單理解就是,拷貝引用,為了避免引用值發(fā)生改變,例如被外部類的方法修改等,而導(dǎo)致內(nèi)部類得到的值不一致,于是用 final 來讓該引用不可改變。

故如果定義了一個(gè)匿名內(nèi)部類,并且希望它使用一個(gè)其外部定義的參數(shù),那么編譯器會(huì)要求該參數(shù)引用是 final 的。

四、匿名內(nèi)部類初始化

我們一般都是利用構(gòu)造器來完成某個(gè)實(shí)例的初始化工作的,但是匿名內(nèi)部類是沒有構(gòu)造器的!那怎么來初始化匿名內(nèi)部類呢?使用構(gòu)造代碼塊!利用構(gòu)造代碼塊能夠達(dá)到為匿名內(nèi)部類創(chuàng)建一個(gè)構(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ù),不懼困難,攀登高峰?。。。。。?/code>

上一篇:fail-fast 機(jī)制下一篇:LinkedList