【java】淺談java內(nèi)部類(二)

局部內(nèi)部類

局部內(nèi)部類是內(nèi)部類的第二種形式,它讓內(nèi)部類的“隱藏”得更深一層——寫在外部類的方法內(nèi)部忆植,而不是處于和外部類方法平行的位置。
讓我們對上面成員內(nèi)部類處理的場景做些思考:我們的Inner內(nèi)部類僅僅只在outterPrint方法中使用了一次:

public void outterPrint () {
  Inner i = new Inner();
  i.innerPrint();
}

那么我們能不能把Inner內(nèi)部類直接定義在outterPrint的內(nèi)部呢?****這樣的話种玛,它就能更好地隱藏起來笑窜,即使是類Outter中除outterPrint外的方法致燥,也不能訪問到它:
現(xiàn)在的Outter的類看起來像這樣:

public class Outter {
  public void outterPrint () {// 外部類方法
    class LocalInner { // 局部內(nèi)部類
      public void innerPrint () {   }
    } 
    LocalInner i = new LocalInner(); // 實例化局部內(nèi)部類
    i.innerPrint();
  }
}

相比于成員內(nèi)部類,局部內(nèi)部類多了一項能訪問的數(shù)據(jù)排截,那就是局部變量(由外部類方法提供)

成員內(nèi)部類:外部類數(shù)據(jù)嫌蚤,內(nèi)部類數(shù)據(jù)
局部內(nèi)部類: 外部類數(shù)據(jù),內(nèi)部類數(shù)據(jù)断傲, 局部數(shù)據(jù)

具體示例如下:

public class Outter {
 private String data = "外部數(shù)據(jù)";  // 外部類數(shù)據(jù)
 public void outterPrint (final String localData) { // 局部數(shù)據(jù)
   class LocalInner {
     private String data = "內(nèi)部數(shù)據(jù)";  // 內(nèi)部類數(shù)據(jù)
     public void innerPrint () {
       System.out.println(Outter.this.data);  // 打印外部類數(shù)據(jù)
       System.out.println(this.data);   //  打印內(nèi)部類數(shù)據(jù)
       System.out.println(localData);  // 打印局部數(shù)據(jù)
     }
   } 
   LocalInner i = new LocalInner();
   i.innerPrint();
 }
}

Test.java:

public class Test {
  public static void main (String [] args) {
    Outter o = new Outter();
    o.outterPrint("局部數(shù)據(jù)");
  }
}

結(jié)果輸出:

外部數(shù)據(jù)
內(nèi)部數(shù)據(jù)
局部數(shù)據(jù)

局部類所使用的外部類方法的形參必須用final修飾

這里要注意一點脱吱, 局部類所使用的外部類方法的形參必須用final修飾,否則會編譯不通過认罩,也就是說傳入后不許改變

為什么這個方法形參一定要用final修飾箱蝠?

** (僅個人理解,如有不同的意見或者更好的理解歡迎在評論區(qū)討論)**

如果不用final修飾會怎樣垦垂? 且聽我慢慢道來:

首先要說一下:

1.內(nèi)部類和外部類在編譯之后形式上是一樣的宦搬,不會有內(nèi)外之分
2.局部內(nèi)部類對于使用的外部方法的值會用構(gòu)造函數(shù)做一個拷貝(編譯后)

例如對于下面outterPrint方法中的LocalInner

public void outterPrint (final String data) {
  class LocalInner {
    public void innerPrint () {
    // 使用 data
    }
  }
}

編譯之后大概長這樣:

public class Outter$LocalInner{ 
  public LocalInner(String data){
    this.LocalInner$data = data; // 對于使用的data做了一次拷貝
  }
  public void innerPrint (){ /* 使用 data */ }
}

這里要注意的是:

1. 編譯后,LocalInner并非直接使用data劫拗,而是用構(gòu)造器拷貝一份后再使用
2. java是值傳遞的间校,所以包裹 LocalInner的外部方法outterPrint也會對傳入的data參數(shù)做一次拷貝(基本類型數(shù)據(jù)拷貝副本,對象等則拷貝引用)

OK页慷,現(xiàn)在的情況是:

方法內(nèi)的局部類對data拷貝了兩次:外部方法outterPrint值傳遞時的拷貝憔足,和LocalInner構(gòu)造函數(shù)的拷貝
方法內(nèi)除了局部類外的作用域只拷貝了data一次: 外部方法outterPrint值傳遞時的拷貝

拷貝兩次和拷貝一次,導(dǎo)致在outterPrint方法內(nèi)部酒繁, 局部類內(nèi)部的data和局部類外部的data是不同步的四瘫! 也即你在局部類內(nèi)部改了data不影響局部類外部的data,在局部類外部改了data也不影響局部類內(nèi)部的data(注意一個前提欲逃,值是基本類型的找蜜,如果是對象的話因為拷貝的是引用仍然可以“同步”)

圖示一:


圖示二:


于是java說: 哎呀媽呀, 這都data都不同步了稳析, 要是讓你修改這還了得!!! 于是就強行要求我們加上final

【注意】所謂的不同步主要是針對基本類型來說的洗做,如果是對象之類的話因為拷貝的是引用所以仍然可以“同步”

如何突破必須用final的限制

我們上面說到,局部內(nèi)部類所使用的方法形參必須用final修飾的限制彰居。

例如

public void outterPrint (String data) {// 沒加上final
  class LocalInner { 
    public void changeData () {
      data = "我想修改data的值";  // 在這一行編譯報錯
    }
   } 
}

提示:

Cannot refer to a non-final variable data inside an inner class defined in a different method

那么诚纸,如果我們有對該形參必須能修改的硬性需求怎么辦?
你可以通過一種有趣的方式繞開它:使用一個單元素數(shù)組陈惰。因為用final修飾的基本類型的變量不允許修改值畦徘,但是卻允許修改final修飾的單元素數(shù)組里的數(shù)組元素, 因為存放數(shù)組的變量的值只是一個引用,我們修改數(shù)組元素的時候是不會修改引用指向的地址的井辆,在這點上final并不會妨礙我們:
Outter.java

public class Outter {
  public void outterPrint (final String []  data) { 
    class LocalInner { 
      public void innerPrint () {
        data[0] = "堂而皇之地修改它9赝病!";   // 修改數(shù)據(jù)
        System.out.print(data[0]);  // 輸出修改后的數(shù)據(jù)
      }
    } 
    LocalInner i = new LocalInner();
    i.innerPrint();
  }
}

Test.java:

public class Test {
  public static void main (String [] args) {
    Outter o = new Outter();
    String [] data = new String [1];
    data[0] = "我是數(shù)據(jù)";
    o.outterPrint(data);  // 修改數(shù)據(jù)并且輸出
  }
}

結(jié)果輸出:

堂而皇之地修改它1薄蒸播!

【注意】局部類不能用public或private訪問符進行聲明!萍肆!

名內(nèi)部類

倘若我們再把局部內(nèi)部類再深化一下袍榆, 那就是匿名內(nèi)部類

匿名內(nèi)部類的使用方式

new [超類/接口] {   /* 類體 */   }

讓我們看看下面這個例子:
Other.java:

public class Other {    }

Outter.java:

public class Outter { public void outterPrint (String data) {    Other o = new Other() {  }; // 匿名內(nèi)部類  }}

何謂之匿名?

誒塘揣,不是說好的匿名嗎包雀? 那么為什么還有個Other的類名呢?”

Other o = new Other() {  /* 匿名內(nèi)部類的類體 */   };

實際上亲铡,這里的Other并不是我們的匿名內(nèi)部類才写,而是我們匿名內(nèi)部類的超類,上面一行代碼其實相當(dāng)于(用成員內(nèi)部類來表示的話)

// annoymous翻譯為匿名
public class Outter {
  private class annoymous extends Other{  }  
  public void outterPrint () { 
    Other a = new annoymous();
  }
}

同時要注意奴愉,我們在使用匿名內(nèi)部類的方式,是在定義一個內(nèi)部類的同時實例化該內(nèi)部類:

new Other() {  /* 匿名內(nèi)部類的類體 */  };  // new操作和定義類的代碼是緊緊結(jié)合在一起的

匿名函數(shù)的作用

用匿名函數(shù)的作用在于在一些特定的場景下寫起來很簡單铁孵,例如事件監(jiān)聽器:

ActionListener listener = new ActionListener() { 
  public void actionPerformed(ActionEvent e) {   }
};

避免了再創(chuàng)建另外一個類文件

講的有點亂锭硼, 對匿名內(nèi)部類做個總結(jié):

  1. 省略被定義的類的類名
  2. 必須結(jié)合超類或者接口使用,即 new [超類/接口] { /* 類體 */ }
  3. 在定義該匿名類的同時實例化該匿名類
  4. 在一些場景下能簡化代碼

【注意】匿名類不能有構(gòu)造器蜕劝, 因為構(gòu)造器和類同名檀头,而匿名類沒有類名,所以匿名類不能有構(gòu)造器

文章總結(jié)

我們使用內(nèi)部類的原因主要有三點:

1.實現(xiàn)數(shù)據(jù)隱藏岖沛, 避免多余的可見性
2.自由訪問外部類的變量

  1. 在使用監(jiān)聽器等場景的時候使用匿名內(nèi)部類暑始,避免增加的大量代碼
    關(guān)于成員內(nèi)部類, 方法局部類婴削,匿名內(nèi)部類的關(guān)系
    從成員內(nèi)部類廊镜,方法局部類到匿名內(nèi)部類是一個不斷深入的關(guān)系, 成員內(nèi)部類進一步隱藏可見性就成為了方法局部類唉俗, 方法局部類省去類名嗤朴,并將類的定義和實例化操作合并到一起,就是匿名內(nèi)部類虫溜。因此雹姊,匿名內(nèi)部類沿襲了成員內(nèi)部類和方法局部類的基本特特
    內(nèi)部類的一些特殊的要求
    1.局部類不能用public或private訪問符進行聲明
    2.局部類所使用的外部類方法的形參必須用final修飾
  2. 匿名內(nèi)部類不能有構(gòu)造器

歡迎加入學(xué)習(xí)交流群569772982,大家一起學(xué)習(xí)交流衡楞。

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末吱雏,一起剝皮案震驚了整個濱河市,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌歧杏,老刑警劉巖镰惦,帶你破解...
    沈念sama閱讀 219,039評論 6 508
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異得滤,居然都是意外死亡陨献,警方通過查閱死者的電腦和手機,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 93,426評論 3 395
  • 文/潘曉璐 我一進店門懂更,熙熙樓的掌柜王于貴愁眉苦臉地迎上來眨业,“玉大人,你說我怎么就攤上這事沮协×浼瘢” “怎么了?”我有些...
    開封第一講書人閱讀 165,417評論 0 356
  • 文/不壞的土叔 我叫張陵慷暂,是天一觀的道長聘殖。 經(jīng)常有香客問我,道長行瑞,這世上最難降的妖魔是什么奸腺? 我笑而不...
    開封第一講書人閱讀 58,868評論 1 295
  • 正文 為了忘掉前任,我火速辦了婚禮血久,結(jié)果婚禮上突照,老公的妹妹穿的比我還像新娘。我一直安慰自己氧吐,他們只是感情好讹蘑,可當(dāng)我...
    茶點故事閱讀 67,892評論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著筑舅,像睡著了一般座慰。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上翠拣,一...
    開封第一講書人閱讀 51,692評論 1 305
  • 那天版仔,我揣著相機與錄音,去河邊找鬼误墓。 笑死邦尊,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的优烧。 我是一名探鬼主播蝉揍,決...
    沈念sama閱讀 40,416評論 3 419
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼畦娄!你這毒婦竟也來了又沾?” 一聲冷哼從身側(cè)響起弊仪,我...
    開封第一講書人閱讀 39,326評論 0 276
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎杖刷,沒想到半個月后励饵,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體,經(jīng)...
    沈念sama閱讀 45,782評論 1 316
  • 正文 獨居荒郊野嶺守林人離奇死亡滑燃,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點故事閱讀 37,957評論 3 337
  • 正文 我和宋清朗相戀三年役听,在試婚紗的時候發(fā)現(xiàn)自己被綠了。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片表窘。...
    茶點故事閱讀 40,102評論 1 350
  • 序言:一個原本活蹦亂跳的男人離奇死亡典予,死狀恐怖,靈堂內(nèi)的尸體忽然破棺而出乐严,到底是詐尸還是另有隱情瘤袖,我是刑警寧澤,帶...
    沈念sama閱讀 35,790評論 5 346
  • 正文 年R本政府宣布昂验,位于F島的核電站捂敌,受9級特大地震影響,放射性物質(zhì)發(fā)生泄漏既琴。R本人自食惡果不足惜占婉,卻給世界環(huán)境...
    茶點故事閱讀 41,442評論 3 331
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望甫恩。 院中可真熱鬧逆济,春花似錦、人聲如沸填物。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,996評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽滞磺。三九已至,卻和暖如春莱褒,著一層夾襖步出監(jiān)牢的瞬間击困,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 33,113評論 1 272
  • 我被黑心中介騙來泰國打工广凸, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留阅茶,地道東北人。 一個月前我還...
    沈念sama閱讀 48,332評論 3 373
  • 正文 我出身青樓谅海,卻偏偏與公主長得像脸哀,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子扭吁,可洞房花燭夜當(dāng)晚...
    茶點故事閱讀 45,044評論 2 355

推薦閱讀更多精彩內(nèi)容

  • 1. Java基礎(chǔ)部分 基礎(chǔ)部分的順序:基本語法撞蜂,類相關(guān)的語法盲镶,內(nèi)部類的語法,繼承相關(guān)的語法蝌诡,異常的語法溉贿,線程的語...
    子非魚_t_閱讀 31,639評論 18 399
  • 一:java概述:1,JDK:Java Development Kit浦旱,java的開發(fā)和運行環(huán)境宇色,java的開發(fā)工...
    ZaneInTheSun閱讀 2,654評論 0 11
  • Java 內(nèi)部類 分四種:成員內(nèi)部類、局部內(nèi)部類颁湖、靜態(tài)內(nèi)部類和匿名內(nèi)部類宣蠕。 1、成員內(nèi)部類: 即作為外部類的一個成...
    ikaroskun閱讀 1,232評論 0 13
  • 什么是版本庫呢爷狈?版本庫又名倉庫植影,可以理解為一個目錄,這個目錄的里面的所有文件都可以被git管理起來涎永,這個目錄下的每...
    Simon_Zhang閱讀 500評論 0 0
  • 有效管理需要充分運用每個人的長處思币,共同完成任務(wù)。用人需先知人而擇人羡微,其基礎(chǔ)為此人能做什么谷饿,即著眼于何所長,不能只想...
    牙印兒小co閱讀 357評論 0 0