Java內部類超詳細總結(含代碼示例)

Java內部類超詳細總結(含代碼示例)
什麼是內部類
什麼是內部類?
顧名思義,就是將一個類的定義放在另一個類的內部。
概念很清楚,感覺很簡單,其實關鍵在於這個內部類放置的位置,可以是一個類的作用域範圍、一個方法的或是一個代碼塊的作用域範圍。
所以理解了概念只是第一步,掌握細節才能徹底搞定Java的內部類特性。
看例子,這是最普通的內部類:

public class Product1 {

class Design{
    private String name = "P30 pro";
    public String showName() {
        return name;
    }
}

class Content{
    private int i;
    Content(int value){
        i = value;
    }
    int value() {return i;}
}
public void show(int value) {
    Content c = new Content(value);
    Design d = new Design();
    System.out.println(d.showName());
    System.out.println(c.value());
}
public static void main(String[] args) {
    Product1 p = new Product1();
    p.show(6000);
}

}
說明:
上面這個示例展示了內部類最基礎的用法,就是將一個或多個類的定義放在了外圍內的內部。可以看到在show()方法中的使用和普通類一樣,沒有區別。

另外,在外圍類的非靜態方法之外的任意位置,不能通過直接new 內部類的方式創建內部類對象。會報編譯錯誤。
像這樣:
在這裏插入圖片描述
還有這樣:
在這裏插入圖片描述
如果想要在外圍類非靜態方法之外創建內部類對象。怎麼辦呢?

正確的姿勢是這樣子:
在外圍類中增加兩個公共方法,返回內部類的引用。

public Design design() {
    return new Design();
}
public Content content(int value) {
    return new Content(value);
}

然後通過外部類對象調用方法的方式獲取內部類對象。

public static void main(String[] args) {
    Product2 p = new Product2();
    p.show(6000);
    Product2.Content c1 = p.content(100);
    Product2.Design d1 = p.design();
}

值得注意的是,外部類之外的其他類,也是不能直接訪問到該外部類的內部類對象的。
會報編譯錯誤。這也符合內部類的定義,就是爲外圍類服務的嘛!

內部類的特性
1.可以鏈接到外部類
當生成一個內部類對象時,此對象與製造它的外圍對象之間就有了一種聯繫,所以它能訪問外圍對象的所有成員,而不需要任何特殊的條件。

下面的例子是《Java編程思想》中的示例
首先定義一個Selector接口

public interface Selector {

boolean end();
Object current();
void next();

}
然後定義一個Sequence類,其中定義有一個private的內部類。

public class Sequence {

private Object[] items;
private int next = 0;
public Sequence(int size) {items = new Object[size];}
public void add(Object x) {
    if(next < items.length) {
        items[next++] = x;
    }
}
private class SequenceSelector implements Selector{
    private int i = 0;
    public boolean end() {
        return i == items.length;
    }
    public Object current() {
        return items[i];
    }
    public void next() {
        if(i < items.length) {i++;}
    }
}
public Selector selector() {
    return new SequenceSelector();
}
public static void main(String[] args) {
    Sequence sequence = new Sequence(10);
    for(int i = 0; i < 10; i++) {
        sequence.add(Integer.toString(i));
    }
    Selector selector = sequence.selector();
    while(!selector.end()) {
        System.out.print(selector.current() + " ");
        selector.next();
    }
}

}
說明:
可以看到SequenceSelector是一個內部類,其end()、current()和next()都用到了外圍類中private的items字段。

2.使用.this和.new
.this的用法
如果需要生成對外部類對象的引用,可以使用外部類的名字後面緊跟圓點和this。
如下所示:

public class DotThis {

void name() {System.out.println("name");}
public class Inner{
    public DotThis outer() {
        return DotThis.this;
    }
}
public Inner inner() {return new Inner();}
public static void main(String[] args) {
    DotThis dt = new DotThis();
    DotThis.Inner inner = dt.inner();
    inner.outer().name();
}

}
需要注意DotThis.this只是產生了正確的外部類引用。並沒有創建外部類對象。

.new的用法
可以通過該語法創建內部類對象,不過要注意的是要使用外部類的對象去創建。

public class DotNew {

public class Inner{}
public static void main(String[] args) {
    DotNew dn = new DotNew();
    DotNew.Inner dnInner = dn.new Inner();
}

}
內部類分類
1、局部內部類
定義在一個方法中或是方法中某一作用域內的類。稱作局部內部類。

public class Product3 {

public Section section(String inputName) {
    class ASection implements Section{
        private String name;
        private ASection(String name) {
            this.name = name;
        }
        @Override
        public String hello() {
            return name + " say hello";
        }
    }
    return new ASection(inputName);
}
public static void main(String[] args) {
    Product3 p = new Product3();
    Section section = p.section("aaaaa");
    System.out.println(section.hello());
}

}
ASection是一個Section接口的實現,Section接口代碼如下:

public interface Section {

String hello();

}
說明:

該內部類在section()方法中,該方法之外不能訪問ASection類;
方法內部類不允許使用訪問權限修飾符(public、private、protected);
注意方法返回的是Section 的引用,即有向上轉型。
另外還可以將內部類的定義放在方法中某一語句塊中,如if語句塊中。

public class Product4 {

public String check(boolean flag) {
    String checkId = null;
    if(flag) {
        class DetailCheck{
            private String id;
            private DetailCheck(String id) {
                this.id = id;
            }
            String getId() {
                return id;
            }
        }
        DetailCheck dc = new DetailCheck("1111");
        checkId = dc.getId();
    }
    return checkId;
}
public static void main(String[] args) {
    Product4 p = new Product4();
    System.out.println(p.check(true));
}

}
說明:
DetailCheck內部類在if語句塊中,因此它的作用範圍也在if語句塊的範圍之內。超出該範圍是不可用的。比如這樣就會編譯報錯:
在這裏插入圖片描述

2、匿名內部類
匿名內部類其實是一種特殊的方法內部類(局部內部類)。它特殊在於將內部類的定義和其對象創建結合在了一塊。沒有通過class關鍵字顯示聲明內部類名稱,故謂之“匿名”。
看代碼示例:

public class Product5 {

public Section section() {
    return new Section() {
        private String name = "hayli";
        @Override
        public String hello() {
            // TODO Auto-generated method stub
            return name + " haha";
        }
    };
}
public static void main(String[] args) {
    Product5 p = new Product5();
    p.section();
}

}
說明:
此處Section可以是接口,也可是基類。

3、嵌套類(靜態內部類)
使用static關鍵字修飾的內部類,叫做靜態內部類,也稱作嵌套類。
嵌套類和普通內部類之間最大的區別是:

普通內部類對象隱式地保存了一個引用,指向創建它的外部類對象。而嵌套類創建對象,並不需要外部類對象。
不能從嵌套類的對象中訪問非靜態的外部類對象。
普通內部類不能有static數據和static字段,也不能包含嵌套類,但是嵌套類可以包含所有這些東西。
public class Product6 {

private static int id = 100;
private static class BSection implements Section{
    private String name = "bbbb";
    @Override
    public String hello() {
        return name + " hello";
    }
    // 只能訪問外部類的靜態數據或字段
    public int getId() {return id;}
    
    // 可以包含靜態數據或方法
    static int x = 200;
    public static void test1() {}
    
    // 可以再嵌套一層
    static class BInner{
        private String name;
        static void test1() {System.out.println("inner ===");}
    }
}
public static void main(String[] args) {
    Section section = new BSection();
    section.hello();
}

總結
本篇介紹了什麼是內部類、內部類最普通定義方法和Java內部類的幾種具體類型詳解。雖然工作中使用內部類的機會不會,但是瞭解這些最基礎的知識,真的在項目中遇到內部類的寫法,也能看懂是怎麼回事了。
原文地址https://www.cnblogs.com/happyone/p/11306419.html

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章