java 嵌套類(lèi)的詳解及實(shí)例代碼
java 嵌套類(lèi)
到現(xiàn)在為止,我們都是在Java文件中直接定義類(lèi)。這樣的類(lèi)出現(xiàn)在包(package)的級(jí)別上。Java允許類(lèi)的嵌套定義。
這里將講解如何在一個(gè)類(lèi)中嵌套定義另一個(gè)類(lèi)。
內(nèi)部類(lèi)
Java允許我們?cè)陬?lèi)的內(nèi)部定義一個(gè)類(lèi)。如果這個(gè)類(lèi)是沒(méi)有static修飾符,那么這樣一個(gè)嵌套在內(nèi)部的類(lèi)稱(chēng)為內(nèi)部類(lèi)(inner class)。
內(nèi)部類(lèi)被認(rèn)為是外部對(duì)象的一個(gè)成員。在定義內(nèi)部類(lèi)時(shí),我們同樣有訪(fǎng)問(wèn)權(quán)限控制(public, private, protected)。
在使用內(nèi)部類(lèi)時(shí),我們要先創(chuàng)建外部對(duì)象。由于內(nèi)部類(lèi)是外部對(duì)象的一個(gè)成員,我們可以在對(duì)象的內(nèi)部自由使用內(nèi)部類(lèi):
實(shí)例代碼:
public class Test { public static void main(String[] args) { Human me = new Human("Vamei"); me.drinkWater(0.3); } } class Human { /** * inner class */ private class Cup { public void useCup(double w) { this.water = this.water - w; } public double getWater() { return this.water; } private double water = 1.0; } /** * constructor */ public Human(String n) { this.myCup = new Cup(); this.name = n; } public void drinkWater(double w) { myCup.useCup(w); System.out.println(myCup.getWater()); } private Cup myCup; private String name; }
上面的例子中,Cup類(lèi)為內(nèi)部類(lèi)。該內(nèi)部類(lèi)有private的訪(fǎng)問(wèn)權(quán)限,因此只能在Human內(nèi)部使用。這樣,Cup類(lèi)就成為一個(gè)被Human類(lèi)專(zhuān)用的類(lèi)。
如果我們使用其他訪(fǎng)問(wèn)權(quán)限,內(nèi)部類(lèi)也能從外部訪(fǎng)問(wèn),比如:
public class Test { public static void main(String[] args) { Human me = new Human("Vamei"); me.drinkWater(0.3); Human.Cup soloCup = me.new Cup(); // be careful here } } class Human { /** * inner class */ class Cup { public void useCup(double w) { this.water = this.water - w; } public double getWater() { return this.water; } private double water = 1.0; } /** * constructor */ public Human(String n) { this.myCup = new Cup(); this.name = n; } public void drinkWater(double w) { myCup.useCup(w); System.out.println(myCup.getWater()); } private Cup myCup; private String name; }
這里,內(nèi)部類(lèi)為默認(rèn)訪(fǎng)問(wèn)權(quán)限(包訪(fǎng)問(wèn)權(quán)限)。我們可以在Test類(lèi)中訪(fǎng)問(wèn)Human的內(nèi)部類(lèi)Cup,并使用該內(nèi)部類(lèi)創(chuàng)建對(duì)象。注意我們創(chuàng)建時(shí)如何說(shuō)明類(lèi)型以及使用new:
Human.Cup soloCup = me.new Cup();
我們?cè)趧?chuàng)建內(nèi)部類(lèi)對(duì)象時(shí),必須基于一個(gè)外部類(lèi)對(duì)象(me),并通過(guò)該外部類(lèi)對(duì)象來(lái)創(chuàng)建Cup對(duì)象(me.new)。我將在下一節(jié)講述其中的含義。
閉包
可以看到,我們直接創(chuàng)建內(nèi)部類(lèi)對(duì)象時(shí),必須是基于一個(gè)外部類(lèi)對(duì)象。也就是說(shuō),內(nèi)部類(lèi)對(duì)象必須依附于某個(gè)外部類(lèi)對(duì)象。
內(nèi)部對(duì)象與外部對(duì)象
與此同時(shí),內(nèi)部類(lèi)對(duì)象可以訪(fǎng)問(wèn)它所依附的外部類(lèi)對(duì)象的成員(即使是private的成員)。從另一個(gè)角度來(lái)說(shuō),內(nèi)部類(lèi)對(duì)象附帶有創(chuàng)建時(shí)的環(huán)境信息,也就是其他語(yǔ)言中的閉包(closure)特性??蓞⒖糚ython閉包
我們看下面的例子:
public class Test { public static void main(String[] args) { Human me = new Human("Vamei"); Human him = new Human("Jerry"); Human.Cup myFirstCup = me.new Cup(); Human.Cup mySecondCup = me.new Cup(); Human.Cup hisCup = him.new Cup(); System.out.println(myFirstCup.whosCup()); System.out.println(mySecondCup.whosCup()); System.out.println(hisCup.whosCup()); } } class Human { /** * inner class */ class Cup { public String whosCup() { return name; // access outer field } } /** * constructor */ public Human(String n) { this.name = n; } public void changeName(String n) { this.name = n; } private String name; }
運(yùn)行結(jié)果:
Vamei Vamei Jerry
在上面的例子中,我們通過(guò)內(nèi)部類(lèi)對(duì)象訪(fǎng)問(wèn)外部類(lèi)對(duì)象的name成員。當(dāng)我們基于不同的外部對(duì)象創(chuàng)建內(nèi)部類(lèi)對(duì)象時(shí),所獲得的環(huán)境信息也將隨之變化。
嵌套static類(lèi)
我們可以在類(lèi)的內(nèi)部定義static類(lèi)。這樣的類(lèi)稱(chēng)為嵌套static類(lèi)(nested static class)。
我們可以直接創(chuàng)建嵌套static類(lèi)的對(duì)象,而不需要依附于外部類(lèi)的某個(gè)對(duì)象。相應(yīng)的,嵌套static類(lèi)也無(wú)法調(diào)用外部對(duì)象的方法,也無(wú)法讀取或修改外部對(duì)象的數(shù)據(jù)。從效果上看,嵌套static類(lèi)拓展了類(lèi)的命名空間(name space),比如下面的Human.Mongolian:
public class Test { public static void main(String[] args) { Human.Mongolian him = new Human.Mongolian(); him.Shout(); } } class Human { /** * nested class */ static class Mongolian { public void Shout() { System.out.println("Oh...Ho..."); } } }
在定義嵌套static類(lèi)時(shí),我們同樣可以有不同的訪(fǎng)問(wèn)權(quán)限修飾符。
總結(jié)
嵌套類(lèi)允許我們更好的組織類(lèi)
內(nèi)部類(lèi)實(shí)現(xiàn)了閉包
感謝閱讀,希望能幫助到大家,謝謝大家對(duì)本站的支持!
相關(guān)文章
Java中StringBuilder常用構(gòu)造方法解析
這篇文章主要介紹了Java中StringBuilder常用構(gòu)造方法解析,StringBuilder是一個(gè)可標(biāo)的字符串類(lèi),我們可以吧它看成是一個(gè)容器這里的可變指的是StringBuilder對(duì)象中的內(nèi)容是可變的,需要的朋友可以參考下2024-01-01重寫(xiě)equals的同時(shí)為何要重寫(xiě)hashCode?
這篇文章主要給大家介紹了關(guān)于重寫(xiě)equals的同時(shí)為何要重寫(xiě)hashCode的相關(guān)資料,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2021-01-01解決Elasticsearch因jdk版本問(wèn)題啟動(dòng)失敗的問(wèn)題
這篇文章主要介紹了解決Elasticsearch因jdk版本問(wèn)題啟動(dòng)失敗的問(wèn)題,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07Java老手該當(dāng)心的13個(gè)錯(cuò)誤
這篇文章主要介紹了Java老手該當(dāng)心的13個(gè)錯(cuò)誤,需要的朋友可以參考下2015-04-04ShardingSphere數(shù)據(jù)分片算法及測(cè)試實(shí)戰(zhàn)
這篇文章主要為大家介紹了ShardingSphere數(shù)據(jù)分片算法及測(cè)試實(shí)戰(zhàn)示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-03-03spring aop execution表達(dá)式的用法
這篇文章主要介紹了spring aop execution表達(dá)式的用法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2021-07-07SpringBoot主鍵ID傳到前端后精度丟失的問(wèn)題解決
這篇文章主要通過(guò)示例為大家詳細(xì)介紹一些SpringBoot如何解決雪花算法主鍵ID傳到前端后精度丟失問(wèn)題,文中的示例代碼講解詳細(xì),需要的可以參考一下2022-05-05Mybatis以main方法形式調(diào)用dao層執(zhí)行代碼實(shí)例
這篇文章主要介紹了Mybatis以main方法形式調(diào)用dao層執(zhí)行代碼實(shí)例,MyBatis 是一款優(yōu)秀的持久層框架,MyBatis 免除了幾乎所有的 JDBC 代碼以及設(shè)置參數(shù)和獲取結(jié)果集的工作,需要的朋友可以參考下2023-08-08