java 內部類 和 匿名內部類

轉自:http://www.cnblogs.com/nerxious/archive/2013/01/24/2875649.html 

和:http://www.cnblogs.com/nerxious/archive/2013/01/25/2876489.html   的內容和評論區



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


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

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

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

實例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在這裏並沒有特殊含義,只是一種表示形式而已


 



因爲In這個內部類被私有了嘛,所以無法通過常規的方式生成內部類對象,只能生成Out類的對象去控制內部類(也就是說,此時只有在Out內部才能實例化或操作In的對象),你知道private權限本身就是僅限此類當中啊。這也是爲什麼普通類不會加private修飾的原因,連看都看不到其內部是啥模樣,更別說生成對象了。不信你可以試試上面的代碼,看能否編譯成功。
總之,內部類挺糾結的,它能隨意訪問外部類屬性,也能隨意被包含它的外部類訪問。


內部類的優點是:內部類可以訪問外部類的私有成員變量,而不需要new外部類的對象。
內部類又分爲:靜態內部類、匿名內部類、局部內部類、成員內部類。

靜態內部類的應用場景是:只可以訪問外部類的靜態成員變量和靜態成員方法。

成員內部類的應用場景是:它可以訪問它的外部類的所有成員變量和方法,不管是靜態的還是非靜態的都可以。

局部內部類:像局部變量一樣,不能被public, protected, private和static修飾。只能訪問方法中定義的final類型的局部變量。

匿名內部類:匿名內部類就是沒有名字的局部內部類,不使用關鍵字class, extends, implements, 沒有構造方法。匿名內部類隱式地繼承了一個父類或者實現了一個接口。匿名內部類使用得比較多,通常是作爲一個方法參數。



java 匿名內部類

匿名內部類也就是沒有名字的內部類

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

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

 

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

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
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:匿名內部類的基本實現

1
2
3
4
5
6
7
8
9
10
11
12
13
14
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接口的匿名內部類實現

1
2
3
4
5
6
7
8
9
10
11
12
13
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();
    }
}

運行結果:1 2 3 4 5

 


終極版本

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

發佈了73 篇原創文章 · 獲贊 1 · 訪問量 7萬+
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章