深入理解Java嵌套類和內(nèi)部類
一、什么是嵌套類及內(nèi)部類
可以在一個(gè)類的內(nèi)部定義另一個(gè)類,這種類稱為嵌套類(nested classes),它有兩種類型:靜態(tài)嵌套類和非靜態(tài)嵌套類。靜態(tài)嵌套類使用很少,最重要的是非靜態(tài)嵌套類,也即是被稱作為內(nèi)部類(inner)。嵌套類從JDK1.1開(kāi)始引入。其中inner類又可分為三種:
其一、在一個(gè)類(外部類)中直接定義的內(nèi)部類;
其二、在一個(gè)方法(外部類的方法)中定義的內(nèi)部類;
其三、匿名內(nèi)部類。
下面,我將說(shuō)明這幾種嵌套類的使用及注意事項(xiàng)。
二、靜態(tài)嵌套類
如下所示代碼為定義一個(gè)靜態(tài)嵌套類,
public class StaticTest { private static String name = "javaJohn"; private String id = "X001"; static class Person{ private String address = "swjtu,chenDu,China"; public String mail = "josserchai@yahoo.com";//內(nèi)部類公有成員 public void display(){ //System.out.println(id);//不能直接訪問(wèn)外部類的非靜態(tài)成員 System.out.println(name);//只能直接訪問(wèn)外部類的靜態(tài)成員 System.out.println("Inner "+address);//訪問(wèn)本內(nèi)部類成員。 } } public void printInfo(){ Person person = new Person(); person.display(); //System.out.println(mail);//不可訪問(wèn) //System.out.println(address);//不可訪問(wèn) System.out.println(person.address);//可以訪問(wèn)內(nèi)部類的私有成員 System.out.println(person.mail);//可以訪問(wèn)內(nèi)部類的公有成員 } public static void main(String[] args) { StaticTest staticTest = new StaticTest(); staticTest.printInfo(); } }
在靜態(tài)嵌套類內(nèi)部,不能訪問(wèn)外部類的非靜態(tài)成員,這是由Java語(yǔ)法中"靜態(tài)方法不能直接訪問(wèn)非靜態(tài)成員"所限定。若想訪問(wèn)外部類的變量,必須通過(guò)其它方法解決,由于這個(gè)原因,靜態(tài)嵌套類使用很少。注意,外部類訪問(wèn)內(nèi)部類的的成員有些特別,不能直接訪問(wèn),但可以通過(guò)內(nèi)部類來(lái)訪問(wèn),這是因?yàn)殪o態(tài)嵌套內(nèi)的所有成員和方法默認(rèn)為靜態(tài)的了。同時(shí)注意,內(nèi)部靜態(tài)類Person只在類StaticTest 范圍內(nèi)可見(jiàn),若在其它類中引用或初始化,均是錯(cuò)誤的。
三、在外部類中定義內(nèi)部類
1、內(nèi)部類分為成員內(nèi)部類、靜態(tài)嵌套類、方法內(nèi)部類、匿名內(nèi)部類。
幾種內(nèi)部類的共性:
A、內(nèi)部類仍然是一個(gè)獨(dú)立的類,在編譯之后會(huì)內(nèi)部類會(huì)被編譯成獨(dú)立的.class文件,但是前面冠以外部類的類命和$符號(hào)。
B、內(nèi)部類不能用普通的方式訪問(wèn)。內(nèi)部類是外部類的一個(gè)成員,因此內(nèi)部類可以自由地訪問(wèn)外部類的成員變量,無(wú)論是否是private的。
如下所示代碼為在外部類中定義兩個(gè)內(nèi)部類及它們的調(diào)用關(guān)系:
public class Outer { int outer_x = 100; class Inner{ public int y = 10; private int z = 9; int m = 5; public void display(){ System.out.println("display outer_x:"+ outer_x); } private void display2(){ System.out.println("display outer_x:"+ outer_x); } } void test(){ Inner inner = new Inner(); inner.display(); inner.display2(); //System.out.println("Inner y:" + y);//不能訪問(wèn)內(nèi)部?jī)?nèi)變量 System.out.println("Inner y:" + inner.y);//可以訪問(wèn) System.out.println("Inner z:" + inner.z);//可以訪問(wèn) System.out.println("Inner m:" + inner.m);//可以訪問(wèn) InnerTwo innerTwo = new InnerTwo(); innerTwo.show(); } class InnerTwo{ Inner innerx = new Inner(); public void show(){ //System.out.println(y);//不可訪問(wèn)Innter的y成員 //System.out.println(Inner.y);//不可直接訪問(wèn)Inner的任何成員和方法 innerx.display();//可以訪問(wèn) innerx.display2();//可以訪問(wèn) System.out.println(innerx.y);//可以訪問(wèn) System.out.println(innerx.z);//可以訪問(wèn) System.out.println(innerx.m);//可以訪問(wèn) } } public static void main(String args[]){ Outer outer = new Outer(); outer.test(); } }
總結(jié):
1、對(duì)于內(nèi)部類,通常在定義類的class關(guān)鍵字前不加public 或 private等限制符,若加了沒(méi)有任何影響。
2、內(nèi)部類中可以直接訪問(wèn)外部類的數(shù)據(jù)成員和方法。
3、另外,就是要注意,內(nèi)部類Inner及InnterTwo只在類Outer的作用域內(nèi)是可知的,如果類Outer外的任何代碼嘗試初始化類Inner或使用它,編譯就不會(huì)通過(guò)。同時(shí),內(nèi)部類的變量成員只在內(nèi)部?jī)?nèi)內(nèi)部可見(jiàn),若外部類或同層次的內(nèi)部類需要訪問(wèn),需采用示例程序中的方法,不可直接訪問(wèn)內(nèi)部類的變量。
四、方法內(nèi)部類
顧名思義,把類放在方法內(nèi)。
class Outer { public void doSomething(){ class Inner{ public void seeOuter(){ } } } }
A、方法內(nèi)部類只能在定義該內(nèi)部類的方法內(nèi)實(shí)例化,不可以在此方法外對(duì)其實(shí)例化。
B、方法內(nèi)部類對(duì)象不能使用該內(nèi)部類所在方法的非final局部變量。
因?yàn)榉椒ǖ木植孔兞课挥跅I?,只存在于該方法的生命期?nèi)。當(dāng)一個(gè)方法結(jié)束,其棧結(jié)構(gòu)被刪除,局部變量成為歷史。但是該方法結(jié)束之后,在方法內(nèi)創(chuàng)建的內(nèi)部類對(duì)象可能仍然存在于堆中!例如,如果對(duì)它的引用被傳遞到其他某些代碼,并存儲(chǔ)在一個(gè)成員變量?jī)?nèi)。
正因?yàn)椴荒鼙WC局部變量的存活期和方法內(nèi)部類對(duì)象的一樣長(zhǎng),所以內(nèi)部類對(duì)象不能使用它們。
下面是完整的例子:
class Outer { public void doSomething(){ final int a =10; class Inner{ public void seeOuter(){ System.out.println(a); } } Inner in = new Inner(); in.seeOuter(); } public static void main(String[] args) { Outer out = new Outer(); out.doSomething(); } }
五、匿名內(nèi)部類
沒(méi)有名字的內(nèi)部類。表面上看起來(lái)它們似乎有名字,實(shí)際那不是它們的名字。
A、繼承式的匿名內(nèi)部類。
class Car { public void drive(){ System.out.println("Driving a car!"); } } class Test{ public static void main(String[] args) { Car car = new Car(){ public void drive(){ System.out.println("Driving another car!"); } }; car.drive(); } }
結(jié)果輸出了:Driving another car! Car引用變量不是引用Car對(duì)象,而是Car匿名子類的對(duì)象。
B、接口式的匿名內(nèi)部類。
interface Vehicle { public void drive(); } class Test{ public static void main(String[] args) { Vehicle v = new Vehicle(){ public void drive(){ System.out.println("Driving a car!"); } }; v.drive(); } }
上面的代碼很怪,好像是在實(shí)例化一個(gè)接口。事實(shí)并非如此,接口式的匿名內(nèi)部類是實(shí)現(xiàn)了一個(gè)接口的匿名類。而且只能實(shí)現(xiàn)一個(gè)接口。
C、參數(shù)式的匿名內(nèi)部類。
class Bar{ void doStuff(Foo f){} } interface Foo{ void foo(); } class Test{ static void go(){ Bar b = new Bar(); b.doStuff(new Foo(){ public void foo(){ System.out.println("foofy"); } }); } }
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
springboot使用AOP+反射實(shí)現(xiàn)Excel數(shù)據(jù)的讀取
本文主要介紹了springboot使用AOP+反射實(shí)現(xiàn)Excel數(shù)據(jù)的讀取,文中通過(guò)示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2022-01-01SpringBoot動(dòng)態(tài)定時(shí)任務(wù)實(shí)現(xiàn)與應(yīng)用詳解
定時(shí)任務(wù)在許多應(yīng)用場(chǎng)景中是必不可少的,特別是在自動(dòng)化任務(wù)執(zhí)行、定期數(shù)據(jù)處理等方面,定時(shí)任務(wù)能極大地提高系統(tǒng)的效率,然而,隨著業(yè)務(wù)需求的變化,定時(shí)任務(wù)的執(zhí)行頻率或時(shí)間點(diǎn)可能需要?jiǎng)討B(tài)調(diào)整,所以本文給大家介紹了SpringBoot動(dòng)態(tài)定時(shí)任務(wù)實(shí)現(xiàn)與應(yīng)用2024-08-08Spring HandlerInterceptor實(shí)現(xiàn)原理代碼解析
這篇文章主要介紹了Spring HandlerInterceptor實(shí)現(xiàn)原理代碼解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下2020-10-10spring cloud consul使用ip注冊(cè)服務(wù)的方法示例
這篇文章主要介紹了spring cloud consul使用ip注冊(cè)服務(wù)的方法示例,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2019-03-03JavaMe開(kāi)發(fā)繪制可自動(dòng)換行文本
JavaMe Graphics類中的drawString不支持文本換行,這樣繪制比較長(zhǎng)的字符串時(shí),文本被繪制在同一行,超過(guò)屏幕部分的字符串被截?cái)嗔?。如何使繪制的文本能自動(dòng)換行呢?2015-09-09