內部類(Nested Class)還是獨立的類,只不過被包含於其餘類中。編譯以後內部類會被編譯成獨立的.class
文件。可是前面冠之外部類的類名和$
符號.html
成員內部類:java
如同一我的是由大腦、肢體、器官等身體結果組成,而內部類至關於其中的某個器官之一,例如心臟:它也有本身的屬性和行爲(血液、跳動)。顯然,此處不能單方面用屬性或者方法表示一個心臟,而須要一個類。而心臟又在人體當中,正如同是內部類在外部內當中。多線程
//外部類 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.class
和Out$In.class
對象
其中$
表明了上面程序中Out.In
中的那個 .
blog
Out.In in = new Out().new In()
能夠用來生成內部類的對象,這種方法存在兩個小知識點須要注意繼承
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
,生不生成外部類對象都不要緊)
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()
生成內部類的對象
也就是說,此時的內部類只有外部類可控制
如同是,個人心臟只能由個人身體控制,其餘人沒法直接訪問它
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
在這裏並無特殊含義,只是一種表示形式而已
匿名內部類也就是沒有名字的內部類
正由於沒有名字,因此匿名內部類只能使用一次,它一般用來簡化代碼編寫
但使用匿名內部類還有個前提條件:必須繼承一個父類或實現一個接口
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類只使用一次,那麼將其編寫爲獨立的一個類豈不是很麻煩?
這個時候就引入了匿名內部類
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中的方法在大括號中實現了
這樣即可以省略一個類的書寫
而且,匿名內部類還能用於接口上
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
接口
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(); } }
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(); } }
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
文件。可是前面冠之外部類的類名和$
符號.
成員內部類:
如同一我的是由大腦、肢體、器官等身體結果組成,而內部類至關於其中的某個器官之一,例如心臟:它也有本身的屬性和行爲(血液、跳動)。顯然,此處不能單方面用屬性或者方法表示一個心臟,而須要一個類。而心臟又在人體當中,正如同是內部類在外部內當中。
//外部類 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.class
和Out$In.class
其中$
表明了上面程序中Out.In
中的那個 .
Out.In in = new Out().new In()
能夠用來生成內部類的對象,這種方法存在兩個小知識點須要注意
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
,生不生成外部類對象都不要緊)
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()
生成內部類的對象
也就是說,此時的內部類只有外部類可控制
如同是,個人心臟只能由個人身體控制,其餘人沒法直接訪問它
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