內部類不是很好理解,但說白了其實也就是一個類中還包含著另外一個類
如同一個人是由大腦、肢體、器官等身體結果組成,而內部類相當于其中的某個器官之一,例如心臟:它也有自己的屬性和行為(血液、跳動)
顯然,此處不能單方面用屬性或者方法表示一個心臟,而需要一個類
而心臟又在人體當中,正如同是內部類在外部內當中
實例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在這里并沒有特殊含義,只是一種表示形式而已
?
6、匿名內部類
?
首先回顧下之前的匿名內部類:
匿名內部類的格式:
new 接口或者接口名(){
重寫方法
};
?本質:是該類或者接口的子類對象
匿名內部類方式使用多線程
1、new Thread(){代碼…}.start();
2、new Thread(new Runnable(){代碼…}).start();
注意://在jdk 1.8以前匿名內部類調用外部類對象的屬性或者方法時外部類需要使用final修飾 如下面代碼所示:
public class Test1 {public void studay(){System.out.println("進入外部類的方法");}//定義一個內部類class Main {public int i = 10;public synchronized void operationSup(){System.out.println("進入內部類的方法");try {i--;System.out.println("Main print i = " + i);Thread.sleep(100);} catch (InterruptedException e) {e.printStackTrace();}}}public static void main(String[] args) {// final SyncException se=new SyncException();final Test1 a=new Test1();//在jdk 1.8以前匿名內部類調用外部類對象的屬性或者方法時外部類需要使用final修飾//定義一個匿名內部類Thread t1 = new Thread(new Runnable() {@Overridepublic void run() {a.studay();}},"t1");t1.start();} }
當:jdk版本換成1.8以下?final Test1 a=new Test1(); 沒有使用final修飾及 Test1 a=new Test();
此時調用外部類方法a.studay();就會報錯 所以此時一個內部類中用外部類中的對象,該對象必須要final修飾才能通過;
就那這個代碼來說,內部類thread在main方法執行完后不會隨著方法被GC回收,而方法中的a(Test1)對象可以看做是一個局部變量,在方法執行完后就會被回收,
而在內部類中卻還有繼續用這個a對象,所以要用final修飾該對象,使其會在內部類中復制一份副本給內部類使用。當然它的值是不會改變的。
?