????:內(nèi)部類不是很好理解,但說白了其實(shí)也就是一個(gè)類中還包含著另外一個(gè)類如同一個(gè)人是由大腦、肢體、器官等身體結(jié)果組成,而內(nèi)部類相當(dāng)于其中的某個(gè)器官之一,例如心臟:它也有自己的屬性和行為(血液、跳動(dòng))顯然,此處不能單方面用屬性或者方法表示一個(gè)心臟,而需要一個(gè)類而心臟又在人體當(dāng)中,正如同是內(nèi)部類在外部內(nèi)當(dāng)中實(shí)例1:內(nèi)部類的基本結(jié)構(gòu)//外部類class Out { &n
內(nèi)部類不是很好理解,但說白了其實(shí)也就是一個(gè)類中還包含著另外一個(gè)類
如同一個(gè)人是由大腦、肢體、器官等身體結(jié)果組成,而內(nèi)部類相當(dāng)于其中的某個(gè)器官之一,例如心臟:它也有自己的屬性和行為(血液、跳動(dòng))
顯然,此處不能單方面用屬性或者方法表示一個(gè)心臟,而需要一個(gè)類
而心臟又在人體當(dāng)中,正如同是內(nèi)部類在外部內(nèi)當(dāng)中
實(shí)例1:內(nèi)部類的基本結(jié)構(gòu)
//外部類class Out { private int age = 12; //內(nèi)部類 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(); */ } }
運(yùn)行結(jié)果:12
從上面的例子不難看出,內(nèi)部類其實(shí)嚴(yán)重破壞了良好的代碼結(jié)構(gòu),但為什么還要使用內(nèi)部類呢?
因?yàn)閮?nèi)部類可以隨意使用外部類的成員變量(包括私有)而不用生成外部類的對(duì)象,這也是內(nèi)部類的唯一優(yōu)點(diǎn)
如同心臟可以直接訪問身體的血液,而不是通過醫(yī)生來抽血
程序編譯過后會(huì)產(chǎn)生兩個(gè).class文件,分別是Out.class和Out$In.class
其中$代表了上面程序中Out.In中的那個(gè) .
Out.In in = new Out().new In()可以用來生成內(nèi)部類的對(duì)象,這種方法存在兩個(gè)小知識(shí)點(diǎn)需要注意
1.開頭的Out是為了標(biāo)明需要生成的內(nèi)部類對(duì)象在哪個(gè)外部類當(dāng)中
2.必須先有外部類的對(duì)象才能生成內(nèi)部類的對(duì)象,因?yàn)閮?nèi)部類的作用就是為了訪問外部類中的成員變量
實(shí)例2:內(nèi)部類中的變量訪問形式
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("內(nèi)部類變量:" + 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(); } }
運(yùn)行結(jié)果:
局部變量:14
內(nèi)部類變量:13
外部類變量:12
從實(shí)例1中可以發(fā)現(xiàn),內(nèi)部類在沒有同名成員變量和局部變量的情況下,內(nèi)部類會(huì)直接訪問外部類的成員變量,而無需指定Out.this.屬性名
否則,內(nèi)部類中的局部變量會(huì)覆蓋外部類的成員變量
而訪問內(nèi)部類本身的成員變量可用this.屬性名,訪問外部類的成員變量需要使用Out.this.屬性名
實(shí)例3:靜態(tài)內(nèi)部類
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(); } }
運(yùn)行結(jié)果:12
可以看到,如果用static 將內(nèi)部內(nèi)靜態(tài)化,那么內(nèi)部類就只能訪問外部類的靜態(tài)成員變量,具有局限性
其次,因?yàn)閮?nèi)部類被靜態(tài)化,因此Out.In可以當(dāng)做一個(gè)整體看,可以直接new 出內(nèi)部類的對(duì)象(通過類名訪問static,生不生成外部類對(duì)象都沒關(guān)系)
實(shí)例4:私有內(nèi)部類
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(); } }
運(yùn)行結(jié)果:12
如果一個(gè)內(nèi)部類只希望被外部類中的方法操作,那么可以使用private聲明內(nèi)部類
上面的代碼中,我們必須在Out類里面生成In類的對(duì)象進(jìn)行操作,而無法再使用Out.In in = new Out().new In() 生成內(nèi)部類的對(duì)象
也就是說,此時(shí)的內(nèi)部類只有外部類可控制
如同是,我的心臟只能由我的身體控制,其他人無法直接訪問它
實(shí)例5:方法內(nèi)部類
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); } }
運(yùn)行結(jié)果:
3
12
在上面的代碼中,我們將內(nèi)部類移到了外部類的方法中,然后在外部類的方法中再生成一個(gè)內(nèi)部類對(duì)象去調(diào)用內(nèi)部類方法
如果此時(shí)我們需要往外部類的方法中傳入?yún)?shù),那么外部類的方法形參必須使用final定義
至于final在這里并沒有特殊含義,只是一種表示形式而已
內(nèi)部類的優(yōu)點(diǎn)是:內(nèi)部類可以訪問外部類的私有成員變量,而不需要new外部類的對(duì)象。
內(nèi)部類又分為:靜態(tài)內(nèi)部類、匿名內(nèi)部類、局部內(nèi)部類、成員內(nèi)部類。
靜態(tài)內(nèi)部類的應(yīng)用場(chǎng)景是:只可以訪問外部類的靜態(tài)成員變量和靜態(tài)成員方法。
成員內(nèi)部類的應(yīng)用場(chǎng)景是:它可以訪問它的外部類的所有成員變量和方法,不管是靜態(tài)的還是非靜態(tài)的都可以。
局部內(nèi)部類:像局部變量一樣,不能被public, protected, private和static修飾。只能訪問方法中定義的final類型的局部變量。
匿名內(nèi)部類:匿名內(nèi)部類就是沒有名字的局部內(nèi)部類,不使用關(guān)鍵字class, extends, implements, 沒有構(gòu)造方法。匿名內(nèi)部類隱式地繼承了一個(gè)父類或者實(shí)現(xiàn)了一個(gè)接口。匿名內(nèi)部類使用得比較多,通常是作為一個(gè)方法參數(shù)。