夯實Java基礎系列18:深刻理解Java內部類及其實現原理

本系列文章將整理到我在GitHub上的《Java面試指南》倉庫,更多精彩內容請到個人倉庫裏查看html

https://github.com/h2pl/Java-Tutorialjava

喜歡的話麻煩點下Star哈android

文章首發於個人我的博客:git

www.how2playlife.comww.how2playlife.com程序員

內部類初探

什麼是內部類?

  內部類是指在一個外部類的內部再定義一個類。內部類做爲外部類的一個成員,而且依附於外部類而存在的。內部類可爲靜態,可用protected和private修飾(而外部類只能使用public和缺省的包訪問權限)。內部類主要有如下幾類:成員內部類、局部內部類、靜態內部類、匿名內部類github

內部類的共性

(1)內部類仍然是一個獨立的類,在編譯以後內部類會被編譯成獨立的.class文件,可是前面冠之外部類的類名和$符號 。面試

(2)內部類不能用普通的方式訪問。算法

(3)內部類聲明成靜態的,就不能隨便的訪問外部類的成員變量了,此時內部類只能訪問外部類的靜態成員變量 。編程

(4)外部類不能直接訪問內部類的的成員,但能夠經過內部類對象來訪問後端

  內部類是外部類的一個成員,所以內部類能夠自由地訪問外部類的成員變量,不管是不是private的。

  由於當某個外圍類的對象建立內部類的對象時,此內部類會捕獲一個隱式引用,它引用了實例化該內部對象的外圍類對象。經過這個指針,能夠訪問外圍類對象的所有狀態。

經過反編譯內部類的字節碼,分析以後主要是經過如下幾步作到的:

  1 編譯器自動爲內部類添加一個成員變量, 這個成員變量的類型和外部類的類型相同, 這個成員變量就是指向外部類對象的引用;

  2 編譯器自動爲內部類的構造方法添加一個參數, 參數的類型是外部類的類型, 在構造方法內部使用這個參數爲1中添加的成員變量賦值;

  3 在調用內部類的構造函數初始化內部類對象時, 會默認傳入外部類的引用。

使用內部類的好處:

靜態內部類的做用:

1 只是爲了下降包的深度,方便類的使用,靜態內部類適用於包含類當中,但又不依賴與外在的類。

2 因爲Java規定靜態內部類不能用使用外在類的非靜態屬性和方法,因此只是爲了方便管理類結構而定義。因而咱們在建立靜態內部類的時候,不須要外部類對象的引用。

非靜態內部類的做用:

1 內部類繼承自某個類或實現某個接口,內部類的代碼操做建立其餘外圍類的對象。因此你能夠認爲內部類提供了某種進入其外圍類的窗口。

2 使用內部類最吸引人的緣由是:每一個內部類都能獨立地繼承自一個(接口的)實現,因此不管外圍類是否已經繼承了某個(接口的)實現,對於內部類都沒有影響

3 若是沒有內部類提供的能夠繼承多個具體的或抽象的類的能力,一些設計與編程問題就很難解決。
從這個角度看,內部類使得多重繼承的解決方案變得完整。接口解決了部分問題,而內部類有效地實現了"多重繼承"。

那靜態內部類與普通內部類有什麼區別呢?

問得好,區別以下:

(1)靜態內部類不持有外部類的引用
在普通內部類中,咱們能夠直接訪問外部類的屬性、方法,即便是private類型也能夠訪問,這是由於內部類持有一個外部類的引用,能夠自由訪問。而靜態內部類,則只能夠訪問外部類的靜態方法和靜態屬性(若是是private權限也能訪問,這是由其代碼位置所決定的),其餘則不能訪問。

(2)靜態內部類不依賴外部類
普通內部類與外部類之間是相互依賴的關係,內部類實例不能脫離外部類實例,也就是說它們會同生同死,一塊兒聲明,一塊兒被垃圾回收器回收。而靜態內部類是能夠獨立存在的,即便外部類消亡了,靜態內部類仍是能夠存在的。

(3)普通內部類不能聲明static的方法和變量
普通內部類不能聲明static的方法和變量,注意這裏說的是變量,常量(也就是final static修飾的屬性)仍是能夠的,而靜態內部類形似外部類,沒有任何限制。

爲何普通內部類不能有靜態變量呢?

1 成員內部類 之因此叫作成員 就是說他是類實例的一部分 而不是類的一部分

2 結構上來講 他和你聲明的成員變量是同樣的地位 一個特殊的成員變量 而靜態的變量是類的一部分和實例無關

3 你若聲明一個成員內部類 讓他成爲主類的實例一部分 而後又想在內部類聲明和實例無關的靜態的東西 你讓JVM情何以堪啊

4 若想在內部類內聲明靜態字段 就必須將其內部類自己聲明爲靜態

非靜態內部類有一個很大的優勢:能夠自由使用外部類的全部變量和方法

下面的例子大概地介紹了

1 非靜態內部類和靜態內部類的區別。

2 不一樣訪問權限的內部類的使用。

3 外部類和它的內部類之間的關係

//本節討論內部類以及不一樣訪問權限的控制
//內部類只有在使用時纔會被加載。
//外部類B
public class B{
    int i = 1;
    int j = 1;
    static int s = 1;
    static int ss = 1;
    A a;
    AA aa;
    AAA aaa;
    //內部類A

    public class A {
//        static void go () {
//
//        }
//        static {
//
//        }
//      static int b = 1;//非靜態內部類不能有靜態成員變量和靜態代碼塊和靜態方法,
        // 由於內部類在外部類加載時並不會被加載和初始化。
        //因此不會進行靜態代碼的調用
        int i = 2;//外部類沒法讀取內部類的成員,而內部類能夠直接訪問外部類成員

        public void test() {
            System.out.println(j);
            j = 2;
            System.out.println(j);
            System.out.println(s);//能夠訪問類的靜態成員變量
        }
        public void test2() {
            AA aa = new AA();
            AAA aaa = new AAA();
        }

    }
    //靜態內部類S,能夠被外部訪問
    public static class S {
        int i = 1;//訪問不到非靜態變量。
        static int s = 0;//能夠有靜態變量

        public static void main(String[] args) {
            System.out.println(s);
        }
        @Test
        public void test () {
//            System.out.println(j);//報錯,靜態內部類不能讀取外部類的非靜態變量
            System.out.println(s);
            System.out.println(ss);
            s = 2;
            ss = 2;
            System.out.println(s);
            System.out.println(ss);
        }
    }

    //內部類AA,其實這裏加protected至關於default
    //由於外部類要調用內部類只能經過B。而且沒法直接繼承AA,因此必須在同包
    //的類中才能調用到(這裏不考慮靜態內部類),那麼就和default同樣了。
    protected class AA{
        int i = 2;//內部類之間不共享變量
        public void test (){
            A a = new A();
            AAA aaa = new AAA();
            //內部類之間能夠互相訪問。
        }
    }
    //包外部依然沒法訪問,由於包沒有繼承關係,因此找不到這個類
    protected static class SS{
        int i = 2;//內部類之間不共享變量
        public void test (){

            //內部類之間能夠互相訪問。
        }
    }
    //私有內部類A,對外不可見,但對內部類和父類可見
    private class AAA {
        int i = 2;//內部類之間不共享變量

        public void test() {
            A a = new A();
            AA aa = new AA();
            //內部類之間能夠互相訪問。
        }
    }
    @Test
    public void test(){
        A a = new A();
        a.test();
        //內部類能夠修改外部類的成員變量
        //打印出 1 2
        B b = new B();

    }
}


//另外一個外部類
class C {
    @Test
    public void test() {
        //首先,其餘類內部類只能經過外部類來獲取其實例。
        B.S s = new B.S();
        //靜態內部類能夠直接經過B類直接獲取,不須要B的實例,和靜態成員變量相似。
        //B.A a = new B.A();
        //當A不是靜態類時這行代碼會報錯。
        //須要使用B的實例來獲取A的實例
        B b = new B();
        B.A a = b.new A();
        B.AA aa = b.new AA();//B和C同包,因此能夠訪問到AA
//      B.AAA aaa = b.new AAA();AAA爲私有內部類,外部類不可見
        //當A使用private修飾時,使用B的實例也沒法獲取A的實例,這一點和私有變量是同樣的。
        //全部普通的內部類與類中的一個變量是相似的。靜態內部類則與靜態成員相似。
    }
}

內部類的加載

可能剛纔的例子中沒辦法直觀地看到內部類是如何加載的,接下來用例子展現一下內部類加載的過程。

1 內部類是延時加載的,也就是說只會在第一次使用時加載。不使用就不加載,因此能夠很好的實現單例模式。

2 不管是靜態內部類仍是非靜態內部類都是在第一次使用時纔會被加載。

3 對於非靜態內部類是不能出現靜態模塊(包含靜態塊,靜態屬性,靜態方法等)

4 非靜態類的使用須要依賴於外部類的對象,詳見上述對象innerClass 的初始化。

簡單來講,類的加載都是發生在類要被用到的時候。內部類也是同樣

1 普通內部類在第一次用到時加載,而且每次實例化時都會執行內部成員變量的初始化,以及代碼塊和構造方法。

2 靜態內部類也是在第一次用到時被加載。可是當它加載完之後就會將靜態成員變量初始化,運行靜態代碼塊,而且只執行一次。固然,非靜態成員和代碼塊每次實例化時也會執行。

總結一下Java類代碼加載的順序,萬變不離其宗。

規律1、初始化構造時,先父後子;只有在父類全部都構造完後子類才被初始化

規律2、類加載先是靜態、後非靜態、最後是構造函數。

靜態構造塊、靜態類屬性按出如今類定義裏面的前後順序初始化,同理非靜態的也是同樣的,只是靜態的只在加載字節碼時執行一次,無論你new多少次,非靜態會在new多少次就執行多少次

規律3、java中的類只有在被用到的時候纔會被加載

規律4、java類只有在類字節碼被加載後才能夠被構形成對象實例

成員內部類

在方法中定義的內部類稱爲局部內部類。與局部變量相似,局部內部類不能有訪問說明符,由於它不是外圍類的一部分,可是它能夠訪問當前代碼塊內的常量,和此外圍類全部的成員。

須要注意的是:
局部內部類只能在定義該內部類的方法內實例化,不能夠在此方法外對其實例化。

public class 局部內部類 {
    class A {//局部內部類就是寫在方法裏的類,只在方法執行時加載,一次性使用。
        public void test() {
            class B {
                public void test () {
                    class C {

                    }
                }
            }
        }
    }
    @Test
    public void test () {
        int i = 1;
        final int j = 2;
        class A {
            @Test
            public void test () {
                System.out.println(i);
                System.out.println(j);
            }
        }
        A a = new A();
        System.out.println(a);
    }

    static class B {
        public static void test () {
            //static class A報錯,方法裏不能定義靜態內部類。
            //由於只有在方法調用時才能進行類加載和初始化。

        }
    }
}

匿名內部類

簡單地說:匿名內部類就是沒有名字的內部類,而且,匿名內部類是局部內部類的一種特殊形式。什麼狀況下須要使用匿名內部類?若是知足下面的一些條件,使用匿名內部類是比較合適的:
只用到類的一個實例。
類在定義後立刻用到。
類很是小(SUN推薦是在4行代碼如下)
給類命名並不會致使你的代碼更容易被理解。
在使用匿名內部類時,要記住如下幾個原則:

1  匿名內部類不能有構造方法。

2  匿名內部類不能定義任何靜態成員、方法和類。

3  匿名內部類不能是public,protected,private,static。

4  只能建立匿名內部類的一個實例。

5 一個匿名內部類必定是在new的後面,用其隱含實現一個接口或實現一個類。

6  因匿名內部類爲局部內部類,因此局部內部類的全部限制都對其生效。

一個匿名內部類的例子:

public class 匿名內部類 {

}
interface D{
    void run ();
}
abstract class E{
    E (){

    }
    abstract void work();
}
class A {

        @Test
        public void test (int k) {
            //利用接口寫出一個實現該接口的類的實例。
            //有且僅有一個實例,這個類沒法重用。
            new Runnable() {
                @Override
                public void run() {
//                    k = 1;報錯,當外部方法中的局部變量在內部類使用中必須改成final類型。
                    //由於方外部法中即便改變了這個變量也不會反映到內部類中。
                    //因此對於內部類來說這只是一個常量。
                    System.out.println(100);
                    System.out.println(k);
                }
            };
            new D(){
                //實現接口的匿名類
                int i =1;
                @Override
                public void run() {
                    System.out.println("run");
                    System.out.println(i);
                    System.out.println(k);
                }
            }.run();
            new E(){
                //繼承抽象類的匿名類
                int i = 1;
                void run (int j) {
                    j = 1;
                }

                @Override
                void work() {

                }
            };
        }

}

匿名內部類裏的final

使用的形參爲什麼要爲final

參考文件:http://android.blog.51cto.com/268543/384844

咱們給匿名內部類傳遞參數的時候,若該形參在內部類中須要被使用,那麼該形參必需要爲final。也就是說:當所在的方法的形參須要被內部類裏面使用時,該形參必須爲final。

爲何必需要爲final呢?

首先咱們知道在內部類編譯成功後,它會產生一個class文件,該class文件與外部類並非同一class文件,僅僅只保留對外部類的引用。當外部類傳入的參數須要被內部類調用時,從java程序的角度來看是直接被調用:

public class OuterClass {
    public void display(final String name,String age){
        class InnerClass{
            void display(){
                System.out.println(name);
            }
        }
    }
}

從上面代碼中看好像name參數應該是被內部類直接調用?其實否則,在java編譯以後實際的操做以下:

public class OuterClass$InnerClass {
    public InnerClass(String name,String age){
        this.InnerClass$name = name;
        this.InnerClass$age = age;
    }
    
    
    public void display(){
        System.out.println(this.InnerClass$name + "----" + this.InnerClass$age );
    }
}

因此從上面代碼來看,內部類並非直接調用方法傳遞的參數,而是利用自身的構造器對傳入的參數進行備份,本身內部方法調用的實際上時本身的屬性而不是外部方法傳遞進來的參數。

直到這裏尚未解釋爲何是final

在內部類中的屬性和外部方法的參數二者從外表上看是同一個東西,但實際上卻不是,因此他們二者是能夠任意變化的,也就是說在內部類中我對屬性的改變並不會影響到外部的形參,而然這從程序員的角度來看這是不可行的。

畢竟站在程序的角度來看這兩個根本就是同一個,若是內部類該變了,而外部方法的形參卻沒有改變這是難以理解和不可接受的,因此爲了保持參數的一致性,就規定使用final來避免形參的不改變。

簡單理解就是,拷貝引用,爲了不引用值發生改變,例如被外部類的方法修改等,而致使內部類獲得的值不一致,因而用final來讓該引用不可改變。

故若是定義了一個匿名內部類,而且但願它使用一個其外部定義的參數,那麼編譯器會要求該參數引用是final的。

內部類初始化

咱們通常都是利用構造器來完成某個實例的初始化工做的,可是匿名內部類是沒有構造器的!那怎麼來初始化匿名內部類呢?使用構造代碼塊!利用構造代碼塊可以達到爲匿名內部類建立一個構造器的效果。

public class OutClass {
    public InnerClass getInnerClass(final int age,final String name){
        return new InnerClass() {
            int age_ ;
            String name_;
            //構造代碼塊完成初始化工做
            {
                if(0 < age && age < 200){
                    age_ = age;
                    name_ = name;
                }
            }
            public String getName() {
                return name_;
            }
            
            public int getAge() {
                return age_;
            }
        };
    }

內部類的重載

  若是你建立了一個內部類,而後繼承其外圍類並從新定義此內部類時,會發生什麼呢?也就是說,內部類能夠被重載嗎?這看起來彷佛是個頗有用的點子,可是「重載」內部類就好像它是外圍類的一個方法,其實並不起什麼做用:

class Egg {
       private Yolk y;
 
       protected class Yolk {
              public Yolk() {
                     System.out.println("Egg.Yolk()");
              }
       }
 
       public Egg() {
              System.out.println("New Egg()");
              y = new Yolk();
       }
}
 
public class BigEgg extends Egg {
       public class Yolk {
              public Yolk() {
                     System.out.println("BigEgg.Yolk()");
              }
       }
 
       public static void main(String[] args) {
              new BigEgg();
       }
}
複製代碼
輸出結果爲:
New Egg()
Egg.Yolk()

缺省的構造器是編譯器自動生成的,這裏是調用基類的缺省構造器。你可能認爲既然建立了BigEgg 的對象,那麼所使用的應該是被「重載」過的Yolk,但你能夠從輸出中看到實際狀況並非這樣的。
這個例子說明,當你繼承了某個外圍類的時候,內部類並無發生什麼特別神奇的變化。這兩個內部類是徹底獨立的兩個實體,各自在本身的命名空間內。

內部類的繼承

由於內部類的構造器要用到其外圍類對象的引用,因此在你繼承一個內部類的時候,事情變得有點複雜。問題在於,那個「祕密的」外圍類對象的引用必須被初始化,而在被繼承的類中並不存在要聯接的缺省對象。要解決這個問題,需使用專門的語法來明確說清它們之間的關聯:

class WithInner {
        class Inner {
                Inner(){
                        System.out.println("this is a constructor in WithInner.Inner");
                };
        }
}
 
public class InheritInner extends WithInner.Inner {
        // ! InheritInner() {} // Won't compile
        InheritInner(WithInner wi) {
                wi.super();
                System.out.println("this is a constructor in InheritInner");
        }
 
        public static void main(String[] args) {
                WithInner wi = new WithInner();
                InheritInner ii = new InheritInner(wi);
        }
}

複製代碼
輸出結果爲:
this is a constructor in WithInner.Inner
this is a constructor in InheritInner

能夠看到,InheritInner 只繼承自內部類,而不是外圍類。可是當要生成一個構造器時,缺省的構造器並不算好,並且你不能只是傳遞一個指向外圍類對象的引用。此外,你必須在構造器內使用以下語法:
enclosingClassReference.super();
這樣才提供了必要的引用,而後程序才能編譯經過。

有關匿名內部類實現回調,事件驅動,委託等機制的文章將在下一節講述。

Java內部類的實現原理

內部類爲何可以訪問外部類的成員?

定義內部類以下:

Java 內部類

使用javap命令進行反編譯。

編譯後獲得Main.class Main$Inner.class兩個文件,反編譯Main$Inner.class文件以下:

Java 內部類

能夠看到,內部類其實擁有外部類的一個引用,在構造函數中將外部類的引用傳遞進來。

匿名內部類爲何只能訪問局部的final變量?

其實能夠這樣想,當方法執行完畢後,局部變量的生命週期就結束了,而局部內部類對象的生命週期可能尚未結束,那麼在局部內部類中訪問局部變量就不可能了,因此將局部變量改成final,改變其生命週期。

編寫代碼以下:

Java 內部類

這段代碼編譯爲Main.class Main$1.class兩個文件,反編譯Main$1.class文件以下:

Java 內部類

能夠看到,java將編譯時已經肯定的值直接複製,進行替換,將沒法肯定的值放到了內部類的常量池中,並在構造函數中將其從常量池取出到字段中。

能夠看出,java將局部變量m直接進行復制,因此其並非原來的值,若在內部類中將m更改,局部變量的m值不會變,就會出現數據不一致,因此java就將其限制爲final,使其不能進行更改,這樣數據不一致的問題就解決了。

參考文章

https://www.cnblogs.com/hujingnb/p/10181621.html

https://blog.csdn.net/codingtu/article/details/79336026

https://www.cnblogs.com/woshimrf/p/java-inner-class.html

https://www.cnblogs.com/dengchengchao/p/9713979.html

微信公衆號

Java技術江湖

若是你們想要實時關注我更新的文章以及分享的乾貨的話,能夠關注個人公衆號【Java技術江湖】一位阿里 Java 工程師的技術小站,做者黃小斜,專一 Java 相關技術:SSM、SpringBoot、MySQL、分佈式、中間件、集羣、Linux、網絡、多線程,偶爾講點Docker、ELK,同時也分享技術乾貨和學習經驗,致力於Java全棧開發!

Java工程師必備學習資源: 一些Java工程師經常使用學習資源,關注公衆號後,後臺回覆關鍵字 「Java」 便可免費無套路獲取。

個人公衆號

我的公衆號:黃小斜

做者是 985 碩士,螞蟻金服 JAVA 工程師,專一於 JAVA 後端技術棧:SpringBoot、MySQL、分佈式、中間件、微服務,同時也懂點投資理財,偶爾講點算法和計算機理論基礎,堅持學習和寫做,相信終身學習的力量!

程序員3T技術學習資源: 一些程序員學習技術的資源大禮包,關注公衆號後,後臺回覆關鍵字 「資料」 便可免費無套路獲取。

相關文章
相關標籤/搜索