1. 程式人生 > >內部類與匿名內部類

內部類與匿名內部類

良好的 運行 實例 基本實現 產生 final 嚴重 代碼 組成

內部類不是很好理解,但說白了其實也就是一個類中還包含著另外一個類

如同一個人是由大腦、肢體、器官等身體結果組成,而內部類相當於其中的某個器官之一,例如心臟:它也有自己的屬性和行為(血液、跳動)

顯然,此處不能單方面用屬性或者方法表示一個心臟,而需要一個類

而心臟又在人體當中,正如同是內部類在外部內當中

 

實例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和Out$In.
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在這裏並沒有特殊含義,只是一種表示形式而已
匿名內部類也就是沒有名字的內部類

正因為沒有名字,所以匿名內部類只能使用一次,它通常用來簡化代碼編寫

但使用匿名內部類還有個前提條件:必須繼承一個父類或實現一個接口

 

實例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類只使用一次,那麽將其編寫為獨立的一個類豈不是很麻煩?

這個時候就引入了匿名內部類

 

實例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中的方法在大括號中實現了

這樣便可以省略一個類的書寫

並且,匿名內部類還能用於接口上

 

實例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

 

由上面的例子可以看出,只要一個類是抽象的或是一個接口,那麽其子類中的方法都可以使用匿名內部類來實現

最常用的情況就是在多線程的實現上,因為要實現多線程必須繼承Thread類或是繼承Runnable接口

 

實例4:Thread類的匿名內部類實現

public class Demo {
    public static void main(String[] args) {
        Thread t = new Thread() {
            public void run() {
                for (int i = 1; i <= 5; i++) {
                    System.out.print(i + " ");
                }
            }
        };
        t.start();
    }
}
運行結果:1 2 3 4 5

 

實例5:Runnable接口的匿名內部類實現

public class Demo {
    public static void main(String[] args) {
        Runnable r = new Runnable() {
            public void run() {
                for (int i = 1; i <= 5; i++) {
                    System.out.print(i + " ");
                }
            }
        };
        Thread t = new Thread(r);
        t.start();
    }
}

內部類與匿名內部類