1. 程式人生 > >六個例項教你正確使用Java內部類(成員內部類、匿名內部類、靜態內部類、區域性內部類)

六個例項教你正確使用Java內部類(成員內部類、匿名內部類、靜態內部類、區域性內部類)

內部類不是很好理解,但說白了其實也就是一個類中還包含著另外一個類
如同一個人是由大腦、肢體、器官等身體結果組成,而內部類相當於其中的某個器官之一,例如心臟:它也有自己的屬性和行為(血液、跳動)
顯然,此處不能單方面用屬性或者方法表示一個心臟,而需要一個類
而心臟又在人體當中,正如同是內部類在外部內當中

例項1:內部類的基本結構

//外部類
class Out {
    private int age = 12;

    //內部類
    class In {
        public void print() {
            System.out.println(age);
        }
    }
}

public
class Demo { public static void main(String[] args) { Out.In in = new Out().new In(); in.print(); //或者採用下種方式訪問 /* Out out = new Out(); Out.In in = out.new In(); in.print(); */ } }

執行結果:12
從上面的例子不難看出,內部類其實嚴重破壞了良好的程式碼結構,但為什麼還要使用內部類呢?
因為內部類可以隨意使用外部類的成員變數(包括私有)而不用生成外部類的物件,這也是內部類的唯一優點
如同心臟可以直接訪問身體的血液,而不是通過醫生來抽

程式編譯過後會產生兩個.class檔案,分別是Out.class和OutIn.class代表了上面程式中Out.In中的那個 .
Out.In in = new Out().new In()可以用來生成內部類的物件,這種方法存在兩個小知識點需要注意
1.開頭的Out是為了標明需要生成的內部類物件在哪個外部類當中
2.必須先有外部類的物件才能生成內部類的物件,因為內部類的作用就是為了訪問外部類中的成員變數

例項2:內部類中的變數訪問形式

class Out {
    private int age = 12;

    class In {
        private
int age = 13; public void print() { int age = 14; System.out.println("區域性變數:" + age); System.out.println("內部類變數:" + this.age); System.out.println("外部類變數:" + Out.this.age); } } } public class Demo { public static void main(String[] args) { Out.In in = new Out().new In(); in.print(); } }

執行結果:
區域性變數:14
內部類變數:13
外部類變數:12
從例項1中可以發現,內部類在沒有同名成員變數和區域性變數的情況下,內部類會直接訪問外部類的成員變數,而無需指定Out.this.屬性名
否則,內部類中的區域性變數會覆蓋外部類的成員變數
而訪問內部類本身的成員變數可用this.屬性名,訪問外部類的成員變數需要使用Out.this.屬性名

例項3:靜態內部類

class Out {
    private static int age = 12;

    static class In {
        public void print() {
            System.out.println(age);
        }
    }
}

public class Demo {
    public static void main(String[] args) {
        Out.In in = new Out.In();
        in.print();
    }
}

執行結果:12
可以看到,如果用static 將內部內靜態化,那麼內部類就只能訪問外部類的靜態成員變數,具有侷限性
其次,因為內部類被靜態化,因此Out.In可以當做一個整體看,可以直接new 出內部類的物件(通過類名訪問static,生不生成外部類物件都沒關係)

例項4:私有內部類

class Out {
    private int age = 12;

    private class In {
        public void print() {
            System.out.println(age);
        }
    }
    public void outPrint() {
        new In().print();
    }
}

public class Demo {
    public static void main(String[] args) {
        //此方法無效
        /*
        Out.In in = new Out().new In();
        in.print();
        */
        Out out = new Out();
        out.outPrint();
    }
}

執行結果:12
如果一個內部類只希望被外部類中的方法操作,那麼可以使用private宣告內部類
上面的程式碼中,我們必須在Out類裡面生成In類的物件進行操作,而無法再使用Out.In in = new Out().new In() 生成內部類的物件
也就是說,此時的內部類只有外部類可控制
如同是,我的心臟只能由我的身體控制,其他人無法直接訪問它

例項5:方法內部類

class Out {
    private int age = 12;

    public void Print(final int x) {
        class In {
            public void inPrint() {
                System.out.println(x);
                System.out.println(age);
            }
        }
        new In().inPrint();
    }
}

public class Demo {
    public static void main(String[] args) {
        Out out = new Out();
        out.Print(3);
    }
}

執行結果:
3
12
在上面的程式碼中,我們將內部類移到了外部類的方法中,然後在外部類的方法中再生成一個內部類物件去呼叫內部類方法
如果此時我們需要往外部類的方法中傳入引數,那麼外部類的方法形參必須使用final定義
至於final在這裡並沒有特殊含義,只是一種表示形式而已
匿名內部類也就是沒有名字的內部類
正因為沒有名字,所以匿名內部類只能使用一次,它通常用來簡化程式碼編寫
但使用匿名內部類還有個前提條件:必須繼承一個父類或實現一個介面

例項6.1:不使用匿名內部類來實現抽象方法

abstract class Person {
    public abstract void eat();
}

class Child extends Person {
    public void eat() {
        System.out.println("eat something");
    }
}

public class Demo {
    public static void main(String[] args) {
        Person p = new Child();
        p.eat();
    }
}

執行結果:eat something
可以看到,我們用Child繼承了Person類,然後實現了Child的一個例項,將其向上轉型為Person類的引用
但是,如果此處的Child類只使用一次,那麼將其編寫為獨立的一個類豈不是很麻煩?
這個時候就引入了匿名內部類

例項6.2:匿名內部類的基本實現

abstract class Person {
    public abstract void eat();
}

public class Demo {
    public static void main(String[] args) {
        Person p = new Person() {
            public void eat() {
                System.out.println("eat something");
            }
        };
        p.eat();
    }
}

執行結果:eat something
可以看到,我們直接將抽象類Person中的方法在大括號中實現了
這樣便可以省略一個類的書寫
並且,匿名內部類還能用於介面上

例項6.3:在介面上使用匿名內部類

interface Person {
    public void eat();
}

public class Demo {
    public static void main(String[] args) {
        Person p = new Person() {
            public void eat() {
                System.out.println("eat something");
            }
        };
        p.eat();
    }
}

執行結果:eat something
由上面的例子可以看出,只要一個類是抽象的或是一個介面,那麼其子類中的方法都可以使用匿名內部類來實現