自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

Java的局部內(nèi)部類以及final類型的參數(shù)和變量

開發(fā) 后端
本文是Thinking In Java中其中一段的閱讀總結(jié)。如果定義一個匿名內(nèi)部類,并且希望它使用一個在其外部定的對象,那么編譯器會要求其參數(shù)引用是final 的。經(jīng)研究,Java虛擬機的實現(xiàn)方式是,編譯器會探測局部內(nèi)部類中是否有直接使用外部定義變量的情況,如果有訪問就會定義一個同類型的變量,然后在構(gòu)造方法中用外部變量給自己定義的變量賦值。

Thinking In Java里面的說法(***正確的說法): 如果定義一個匿名內(nèi)部類,并且希望它使用一個在其外部定的對象,那么編譯器會要求其參數(shù)引用是final 的。

  1. public class Tester {     
  2.     public static void main(String[] args) {     
  3.         A a = new A();     
  4.         C c = new C();     
  5.         c.shoutc(a.shout(5));     
  6.     }     
  7. }     
  8. ////////////////////////////////////////////////////////     
  9. class A {     
  10.     public void shouta() {     
  11.         System.out.println("Hello A");     
  12.     }     
  13.     
  14.     public A shout(final int arg) {     
  15.         class B extends A {     
  16.             public void shouta() {     
  17.                 System.out.println("Hello B" + arg);     
  18.             }     
  19.         }     
  20.         return new B();     
  21.     }     
  22. }     
  23. ////////////////////////////////////////////////////////     
  24. class C {     
  25.     void shoutc(A a) {     
  26.         a.shouta();     
  27.     }     
  28. }   

第5行c.shoutc(a.shout(5)),在a.shout(5)得到返回值后,a的shout()方法棧被清空了,即arg不存在了,而c.shoutc()卻又調(diào)用了a.shouta()去執(zhí)行System.out.println("Hello B" + arg)。

再來看Java虛擬機是怎么實現(xiàn)這個詭異的訪問的:有人認為這種訪問之所以能完成,是因為arg是final的,由于變量的生命周期,事實是這樣的嗎?方法棧都不存在了,變量即使存在,怎么可能還被訪問到?試想下:一個方法能訪問另一個方法的定義的final局部變量嗎(不通過返回值)?

研究一下這個詭異的訪問執(zhí)行的原理,用反射探測一下局部內(nèi)部類 。編譯器會探測局部內(nèi)部類中是否有直接使用外部定義變量的情況,如果有訪問就會定義一個同類型的變量,然后在構(gòu)造方法中用外部變量給自己定義的變量賦值,而后局部內(nèi)部類所使用的變量都是自己定義的變量,所以就可以訪問了。見下:

  1. class   A$1$B   
  2. {   
  3. A$1$B(A,   int);   
  4.  
  5. private   final   int   var$arg;   
  6. private   final   A   this$0;   
  7. }   

A$1$B類型的對象會使用自定義的var$arg變量,而不是shout()方法中的final int arg變量,當(dāng)然就可以訪問了。

那么為什么外部變量要是final的呢?即使外部變量不是final,編譯器也可以如此處理:自己定義一個同類型的變量,然后在構(gòu)造方法中賦值就行了。原因就是為了讓我們能夠挺合邏輯的直接使用外部變量,而且看起來是在始終使用 外部的arg變量(而不是賦值以后的自己的字段)。

考慮出現(xiàn)這種情況:在局部內(nèi)部類中使用外部變量arg,如果編譯器允許arg不是final的,那么就可以對這個變量作變值操作(例如arg++),根據(jù)前面的分析,變值操作改變的是var$arg,而外部的變量arg并沒有變,仍然是5(var$arg才是6)。因此為了避免這樣如此不合邏輯的事情發(fā)生:你用了外部變量,又改變了變量的值,但那個變量卻沒有變化,自然的arg就被強行規(guī)定必須是final所修飾的,以確保讓兩個值永遠一樣,或所指向的對象永遠一樣(后者可能更重要)。

還有一點需要注意的是內(nèi)部類與方法不是同時執(zhí)行的,比如實現(xiàn)ActionListener,只有當(dāng)事件發(fā)生的時候才會執(zhí)行,而這時方法已經(jīng)結(jié)束了。

【編輯推薦】

  1. 沒有原生數(shù)據(jù)類型,Java會更好嗎?
  2. 20個開發(fā)人員非常有用的Java功能代碼
  3. 走進Java 7中的模塊系統(tǒng)
  4. 2009年十大Java技術(shù)解決方案
  5. 2008最值得學(xué)習(xí)的五種JAVA技術(shù)
責(zé)任編輯:yangsai 來源: JavaEye
相關(guān)推薦

2020-01-15 11:14:21

Java算法排序

2011-11-23 10:59:18

Javafinal

2009-08-18 17:17:05

C#局部類型

2020-12-14 10:23:23

Java內(nèi)部類外部類

2009-06-11 13:08:29

Java內(nèi)部類Java編程思想

2011-07-21 15:44:33

Java內(nèi)部類

2023-10-19 13:24:00

Java工具

2015-12-08 09:05:41

Java內(nèi)部類

2009-07-29 09:18:49

Java內(nèi)部類

2009-09-17 13:05:38

Linq局部變量類型

2011-03-29 14:11:15

內(nèi)部類

2011-12-06 11:12:59

Java

2023-03-06 07:53:36

JavaN種內(nèi)部類

2010-02-05 15:32:33

Java內(nèi)部類

2010-01-28 15:22:12

C++嵌套類

2009-09-11 10:07:05

Linq隱式類型化局部

2019-12-23 14:32:38

Java內(nèi)部類代碼

2009-08-27 10:08:36

C#隱含類型局部變量

2018-05-14 09:15:24

Python變量函數(shù)

2010-04-29 09:52:27

Oracle鎖
點贊
收藏

51CTO技術(shù)棧公眾號