Java 內部類

內部類(Nested Class)還是獨立的類,只不過被包含於其餘類中。編譯以後內部類會被編譯成獨立的.class文件。可是前面冠之外部類的類名和$符號.html

成員內部類:java

  1. 便於訪問外部類屬性。
  2. 內部類輔助外部類完成某些功能。

如同一我的是由大腦、肢體、器官等身體結果組成,而內部類至關於其中的某個器官之一,例如心臟:它也有本身的屬性和行爲(血液、跳動)。顯然,此處不能單方面用屬性或者方法表示一個心臟,而須要一個類。而心臟又在人體當中,正如同是內部類在外部內當中。多線程

實例1:內部類的基本結構

//外部類
class Out {
    private int age = 12;
     
    //內部類
    class In {
        public void print() {
            System.out.println(age);
        }
    }
}
 
public class Demo {
    public static void main(String[] args) {
        Out.In in = new Out().new In();
        in.print();
        //或者採用下種方式訪問
        /*
        Out out = new Out();
        Out.In in = out.new In();
        in.print();
        */
    }
}

運行結果:12this

從上面的例子不難看出,內部類其實嚴重破壞了良好的代碼結構,但爲何還要使用內部類呢?線程

由於內部類能夠隨意使用外部類的成員變量(包括私有)而不用生成外部類的對象,這也是內部類的惟一優勢code

如同心臟能夠直接訪問身體的血液,而不是經過醫生來抽血htm

程序編譯事後會產生兩個.class文件,分別是Out.classOut$In.class對象

其中$表明了上面程序中Out.In中的那個 .blog

Out.In in = new Out().new In()能夠用來生成內部類的對象,這種方法存在兩個小知識點須要注意繼承

  • 1.開頭的Out是爲了標明須要生成的內部類對象在哪一個外部類當中
  • 2.必須先有外部類的對象才能生成內部類的對象,由於內部類的做用就是爲了訪問外部類中的成員變量

實例2:內部類中的變量訪問形式

class Out {
    private int age = 12;
     
    class In {
        private int age = 13;
        public void print() {
            int age = 14;
            System.out.println("局部變量:" + age);
            System.out.println("內部類變量:" + this.age);
            System.out.println("外部類變量:" + Out.this.age);
        }
    }
}
 
public class Demo {
    public static void main(String[] args) {
        Out.In in = new Out().new In();
        in.print();
    }
}

運行結果

局部變量:14
內部類變量:13
外部類變量:12

從實例1中能夠發現,內部類在沒有同名成員變量和局部變量的狀況下,內部類會直接訪問外部類的成員變量,而無需指定Out.this.屬性名

不然,內部類中的局部變量會覆蓋外部類的成員變量

而訪問內部類自己的成員變量可用this.屬性名,訪問外部類的成員變量須要使用Out.this.屬性名

靜態內部類

class Out {
    private static int age = 12;
     
    static class In {
        public void print() {
            System.out.println(age);
        }
    }
}
 
public class Demo {
    public static void main(String[] args) {
        Out.In in = new Out.In();
        in.print();
    }
}

能夠看到,若是用static 將內部內靜態化,那麼內部類就只能訪問外部類的靜態成員變量,具備侷限性

其次,由於內部類被靜態化,所以Out.In能夠當作一個總體看,能夠直接new 出內部類的對象(經過類名訪問static,生不生成外部類對象都不要緊)

實例4:私有內部類

class Out {
    private int age = 12;
     
    private class In {
        public void print() {
            System.out.println(age);
        }
    }
    public void outPrint() {
        new In().print();
    }
}
 
public class Demo {
    public static void main(String[] args) {
        //此方法無效
        /*
        Out.In in = new Out().new In();
        in.print();
        */
        Out out = new Out();
        out.outPrint();
    }
}

若是一個內部類只但願被外部類中的方法操做,那麼可使用private聲明內部類

上面的代碼中,咱們必須在Out類裏面生成In類的對象進行操做,而沒法再使用Out.In in = new Out().new In() 生成內部類的對象

也就是說,此時的內部類只有外部類可控制

如同是,個人心臟只能由個人身體控制,其餘人沒法直接訪問它

實例5:方法內部類

class Out {
    private int age = 12;
 
    public void Print(final int x) {
        class In {
            public void inPrint() {
                System.out.println(x);
                System.out.println(age);
            }
        }
        new In().inPrint();
    }
}
 
public class Demo {
    public static void main(String[] args) {
        Out out = new Out();
        out.Print(3);
    }
}

在上面的代碼中,咱們將內部類移到了外部類的方法中,而後在外部類的方法中再生成一個內部類對象去調用內部類方法

若是此時咱們須要往外部類的方法中傳入參數,那麼外部類的方法形參必須使用final定義

至於final在這裏並無特殊含義,只是一種表示形式而已

匿名內部類

匿名內部類也就是沒有名字的內部類

正由於沒有名字,因此匿名內部類只能使用一次,它一般用來簡化代碼編寫

但使用匿名內部類還有個前提條件:必須繼承一個父類或實現一個接口

實例1:不使用匿名內部類來實現抽象方法

abstract class Person {
    public abstract void eat();
}
 
class Child extends Person {
    public void eat() {
        System.out.println("eat something");
    }
}
 
public class Demo {
    public static void main(String[] args) {
        Person p = new Child();
        p.eat();
    }
}

運行結果eat something

能夠看到,咱們用Child繼承了Person類,而後實現了Child的一個實例,將其向上轉型爲Person類的引用

可是,若是此處的Child類只使用一次,那麼將其編寫爲獨立的一個類豈不是很麻煩?

這個時候就引入了匿名內部類

實例2:匿名內部類的基本實現

abstract class Person {
    public abstract void eat();
}
 
public class Demo {
    public static void main(String[] args) {
        Person p = new Person() {
            public void eat() {
                System.out.println("eat something");
            }
        };
        p.eat();
    }
}

運行結果eat something

能夠看到,咱們直接將抽象類Person中的方法在大括號中實現了

這樣即可以省略一個類的書寫

而且,匿名內部類還能用於接口上

實例3:在接口上使用匿名內部類

interface Person {
    public void eat();
}
 
public class Demo {
    public static void main(String[] args) {
        Person p = new Person() {
            public void eat() {
                System.out.println("eat something");
            }
        };
        p.eat();
    }
}

由上面的例子能夠看出,只要一個類是抽象的或是一個接口,那麼其子類中的方法均可以使用匿名內部類來實現

最經常使用的狀況就是在多線程的實現上,由於要實現多線程必須繼承Thread類或是繼承Runnable接口

實例4:Thread類的匿名內部類實現

public class Demo {
    public static void main(String[] args) {
        Thread t = new Thread() {
            public void run() {
                for (int i = 1; i <= 5; i++) {
                    System.out.print(i + " ");
                }
            }
        };
        t.start();
    }
}

實例5:Runnable接口的匿名內部類實現

public class Demo {
    public static void main(String[] args) {
        Runnable r = new Runnable() {
            public void run() {
                for (int i = 1; i <= 5; i++) {
                    System.out.print(i + " ");
                }
            }
        };
        Thread t = new Thread(r);
        t.start();
    }
}

實例6:運用內部類調用其餘包類中的protected 的方法

當你想使用一個類的protected 方法時,可是又不和這個類在同一個包下,你是沒辦法調用的。 這時候匿名類就派上用場了,你能夠聲明一個匿名類繼承該類,並定義一個方法,在這個方法內使用super調用你想調用的那個方法(其實你也能夠寫個類繼承這個類,就能調用父類的protected方法了,可是匿名類更簡潔,由於你只想調用這個方法而已)。例如:

public class Testclass {
	protected void test(){
		System.out.println("test...");
	}
}

這個類有一個protected方法test,若是你在其餘包下想調用這個protected方法是不行的

這時候你可使用匿名類繼承這個類,定義一個方法callParentTest(),在這個方法體內調用super.test(),最後調用這個callParentTest()便可。

public class UserOthreMethod{
    public static void main(String[] args){
        new Testclass{
            void callParentTest(){
                super.test();
            }
        }.callParentTest();
    }
}

內部類(Nested Class)還是獨立的類,只不過被包含於其餘類中。編譯以後內部類會被編譯成獨立的.class文件。可是前面冠之外部類的類名和$符號.

成員內部類:

  1. 便於訪問外部類屬性。
  2. 內部類輔助外部類完成某些功能。

如同一我的是由大腦、肢體、器官等身體結果組成,而內部類至關於其中的某個器官之一,例如心臟:它也有本身的屬性和行爲(血液、跳動)。顯然,此處不能單方面用屬性或者方法表示一個心臟,而須要一個類。而心臟又在人體當中,正如同是內部類在外部內當中。

實例1:內部類的基本結構

//外部類
class Out {
    private int age = 12;
     
    //內部類
    class In {
        public void print() {
            System.out.println(age);
        }
    }
}
 
public class Demo {
    public static void main(String[] args) {
        Out.In in = new Out().new In();
        in.print();
        //或者採用下種方式訪問
        /*
        Out out = new Out();
        Out.In in = out.new In();
        in.print();
        */
    }
}

運行結果:12

從上面的例子不難看出,內部類其實嚴重破壞了良好的代碼結構,但爲何還要使用內部類呢?

由於內部類能夠隨意使用外部類的成員變量(包括私有)而不用生成外部類的對象,這也是內部類的惟一優勢

如同心臟能夠直接訪問身體的血液,而不是經過醫生來抽血

程序編譯事後會產生兩個.class文件,分別是Out.classOut$In.class

其中$表明了上面程序中Out.In中的那個 .

Out.In in = new Out().new In()能夠用來生成內部類的對象,這種方法存在兩個小知識點須要注意

  • 1.開頭的Out是爲了標明須要生成的內部類對象在哪一個外部類當中
  • 2.必須先有外部類的對象才能生成內部類的對象,由於內部類的做用就是爲了訪問外部類中的成員變量

實例2:內部類中的變量訪問形式

class Out {
    private int age = 12;
     
    class In {
        private int age = 13;
        public void print() {
            int age = 14;
            System.out.println("局部變量:" + age);
            System.out.println("內部類變量:" + this.age);
            System.out.println("外部類變量:" + Out.this.age);
        }
    }
}
 
public class Demo {
    public static void main(String[] args) {
        Out.In in = new Out().new In();
        in.print();
    }
}

運行結果

局部變量:14
內部類變量:13
外部類變量:12

從實例1中能夠發現,內部類在沒有同名成員變量和局部變量的狀況下,內部類會直接訪問外部類的成員變量,而無需指定Out.this.屬性名

不然,內部類中的局部變量會覆蓋外部類的成員變量

而訪問內部類自己的成員變量可用this.屬性名,訪問外部類的成員變量須要使用Out.this.屬性名

靜態內部類

class Out {
    private static int age = 12;
     
    static class In {
        public void print() {
            System.out.println(age);
        }
    }
}
 
public class Demo {
    public static void main(String[] args) {
        Out.In in = new Out.In();
        in.print();
    }
}

能夠看到,若是用static 將內部內靜態化,那麼內部類就只能訪問外部類的靜態成員變量,具備侷限性

其次,由於內部類被靜態化,所以Out.In能夠當作一個總體看,能夠直接new 出內部類的對象(經過類名訪問static,生不生成外部類對象都不要緊)

實例4:私有內部類

class Out {
    private int age = 12;
     
    private class In {
        public void print() {
            System.out.println(age);
        }
    }
    public void outPrint() {
        new In().print();
    }
}
 
public class Demo {
    public static void main(String[] args) {
        //此方法無效
        /*
        Out.In in = new Out().new In();
        in.print();
        */
        Out out = new Out();
        out.outPrint();
    }
}

若是一個內部類只但願被外部類中的方法操做,那麼可使用private聲明內部類

上面的代碼中,咱們必須在Out類裏面生成In類的對象進行操做,而沒法再使用Out.In in = new Out().new In() 生成內部類的對象

也就是說,此時的內部類只有外部類可控制

如同是,個人心臟只能由個人身體控制,其餘人沒法直接訪問它

實例5:方法內部類

class Out {
    private int age = 12;
 
    public void Print(final int x) {
        class In {
            public void inPrint() {
                System.out.println(x);
                System.out.println(age);
            }
        }
        new In().inPrint();
    }
}
 
public class Demo {
    public static void main(String[] args) {
        Out out = new Out();
        out.Print(3);
    }
}

在上面的代碼中,咱們將內部類移到了外部類的方法中,而後在外部類的方法中再生成一個內部類對象去調用內部類方法

若是此時咱們須要往外部類的方法中傳入參數,那麼外部類的方法形參必須使用final定義

至於final在這裏並無特殊含義,只是一種表示形式而已

資料 http://www.cnblogs.com/nerxious/archive/2013/01/24/2875649.html https://www.zhihu.com/question/49330534

相關文章
相關標籤/搜索