內部類不是很好理解,但說白了其實也就是一個類中還包含著另外一個類
如同一個人是由大腦、肢體、器官等身體結果組成,而內部類相當於其中的某個器官之一,例如心髒:它也有自己的屬性和行為(血液、跳動)
顯然,此處不能單方面用屬性或者方法表示一個心髒,而需要一個類
而心髒又在人體當中,正如同是內部類在外部內當中
//外部類 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.必須先有外部類的對象才能生成內部類的對象,因為內部類的作用就是為了訪問外部類中的成員變量
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.屬性名
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,生不生成外部類對象都沒關系)
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() 生成內部類的對象
也就是說,此時的內部類只有外部類可控制
如同是,我的心髒只能由我的身體控制,其他人無法直接訪問它
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在這裡並沒有特殊含義,只是一種表示形式而已
內部類的優點是:內部類可以訪問外部類的私有成員變量,而不需要new外部類的對象。
內部類又分為:靜態內部類、匿名內部類、局部內部類、成員內部類。
靜態內部類的應用場景是:只可以訪問外部類的靜態成員變量和靜態成員方法。
成員內部類的應用場景是:它可以訪問它的外部類的所有成員變量和方法,不管是靜態的還是非靜態的都可以。
局部內部類:像局部變量一樣,不能被public, protected, private和static修飾。只能訪問方法中定義的final類型的局部變量。
匿名內部類:匿名內部類就是沒有名字的局部內部類,不使用關鍵字class, extends, implements, 沒有構造方法。匿名內部類隱式地繼承了一個父類或者實現了一個接口。匿名內部類使用得比較多,通常是作為一個方法參數。