把類定義在另外一個類的內部,該類就被稱爲內部類。java
舉例:把類Inner定義在類Outer中,類Inner就被稱爲內部類。安全
class Outer { class Inner { } }
A:能夠直接訪問外部類的成員,包括私有this
B:外部類要想訪問內部類成員,必須建立對象spa
A:成員內部類code
B:局部內部類對象
C:靜態內部類繼承
D:匿名內部類接口
成員內部類——就是位於外部類成員位置的類特色:可使用外部類中全部的成員變量和成員方法(包括private的)內存
class Outer { private int age = 20; //成員位置 class Inner { public void show() { System.out.println(age); } } } class Test { public static void main(String[] ages) { //成員內部類是非靜態的演示 Outer.Inner oi = new Outer.new Inner(); oi.show(); } }
//成員內部類不是靜態的: 外部類名.內部類名 對象名 = new 外部類名.new 內部類名(); //成員內部類是靜態的: 外部類名.內部類名 對象名 = new 外部類名.內部類名();
A:private作用域
若是咱們的內部類不想輕易被任何人訪問,能夠選擇使用private修飾內部類,這樣咱們就沒法經過建立對象的方法來訪問,想要訪問只須要在外部類中定義一個public修飾的方法,間接調用。這樣作的好處就是,咱們能夠在這個public方法中增長一些判斷語句,起到數據安全的做用。
class Outer { private class Inner { public void show() { System.out.println(「密碼備份文件」); } } public void method() { if(你是管理員){ Inner i = new Inner(); i.show(); }else { System.out.println(「你沒有權限訪問」); } } }
下面咱們給出一個更加規範的寫法
class Outer { private class Inner { public void show() { System.out.println(「密碼備份文件」); } } //使用getXxx()獲取成員內部類,能夠增長校驗語句(文中省略) public void getInner() { return new Inner(); } public static void main(String[] args) { Outer outer = new Outer(); Outer.Inner inner = outer.getInner(); inner.show(); } }
這種被 static 所修飾的內部類,按位置分,屬於成員內部類,但也能夠稱做靜態內部類,也常叫作嵌套內部類。具體內容咱們在下面詳細講解。
D:成員內部類經典題(填空)
請在三個println 後括號中填空使得輸出25,20,18
class Outer { public int age = 18; class Inner { public int age = 20; public viod showAge() { int age = 25; System.out.println(age);//空1 System.out.println(this.age);//空2 System.out.println(Outer.this.age);//空3 } } }
局部內部類——就是定義在一個方法或者一個做用域裏面的類特色:主要是做用域發生了變化,只能在自身所在方法和屬性中被使用
A 格式:
class Outer { public void method(){ class Inner { } } }
B:訪問時:
//在局部位置,能夠建立內部類對象,經過對象調用和內部類方法 class Outer { private int age = 20; public void method() { final int age2 = 30; class Inner { public void show() { System.out.println(age); //從內部類中訪問方法內變量age2,須要將變量聲明爲最終類型。 System.out.println(age2); } } Inner i = new Inner(); i.show(); } }
C: 爲何局部內部類訪問局部變量必須加final修飾呢?
由於局部變量是隨着方法的調用而調用,使用完畢就消失,而堆內存的數據並不會當即消失。
因此,堆內存仍是用該變量,而該變量已經沒有了。爲了讓該值還存在,就加final修飾。
緣由是,當咱們使用final修飾變量後,堆內存直接存儲的是值,而不是變量名。
(即上例 age2 的位置存儲着常量30 而不是 age2 這個變量名)
咱們所知道static是不能用來修飾類的,可是成員內部類能夠看作外部類中的一個成員,因此能夠用static修飾,這種用static修飾的內部類咱們稱做靜態內部類,也稱做嵌套內部類.特色:不能使用外部類的非static成員變量和成員方法
解釋:非靜態內部類編譯後會默認的保存一個指向外部類的引用,而靜態類卻沒有。
簡單理解:
即便沒有外部類對象,也能夠建立靜態內部類對象,而外部類的非static成員必須依賴於對象的調用,靜態成員則能夠直接使用類調用,沒必要依賴於外部類的對象,因此靜態內部類只能訪問靜態的外部屬性和方法。
class Outter { int age = 10; static age2 = 20; public Outter() { } static class Inner { public method() { System.out.println(age);//錯誤 System.out.println(age2);//正確 } } } public class Test { public static void main(String[] args) { Outter.Inner inner = new Outter.Inner(); inner.method(); } }
一個沒有名字的類,是內部類的簡化寫法
A 格式:
new 類名或者接口名() { 重寫方法(); }
本質:實際上是繼承該類或者實現接口的子類匿名對象
這也就是下例中,能夠直接使用 new Inner() {}.show(); 的緣由 == 子類對象.show();
interface Inter { public abstract void show(); } class Outer { public void method(){ new Inner() { public void show() { System.out.println("HelloWorld"); } }.show(); } } class Test { public static void main(String[] args) { Outer o = new Outer(); o.method(); } }
若是匿名內部類中有多個方法又該如何調用呢?
Inter i = new Inner() { //多態,由於new Inner(){}表明的是接口的子類對象 public void show() { System.out.println("HelloWorld"); } };
B:匿名內部類在開發中的使用
咱們在開發的時候,會看到抽象類,或者接口做爲參數。
而這個時候,實際須要的是一個子類對象。
若是該方法僅僅調用一次,咱們就可使用匿名內部類的格式簡化。
若是內容中有什麼不足,或者錯誤的地方,歡迎你們給我留言提出意見, 蟹蟹你們 !^_^
若是能幫到你的話,那就來關注我吧!(系列文章均會在公衆號第一時間更新)
在這裏的咱們素不相識,卻都在爲了本身的夢而努力 ❤一個堅持推送原創Java技術的公衆號:理想二旬不止