Java內部類學習總結
目錄
概述
最近學習python,發現python是支援多繼承的,這讓我想起Java是通過內部類實現的這套機制。這篇文章不是講如何通過內部類實現多繼承,而是總結一下內部類的型別和使用方法。
Java內部類分為:
- 非靜態內部類
- 靜態內部類
- 區域性內部類
- 匿名內部類
內部類在Android原始碼中被大量的使用,先介紹一下這四種內部類的共同點:
- 內部類仍然是一個獨立的類,在編譯之後內部類會被編譯成獨立的.class檔案,但是前面冠以外部類的類名和$符號。
- 內部類不能用普通的方式訪問。內部類是外部類的一個成員,因為內部類可以自由的訪問外部類的成員,包括private成員。
- 內部類宣告為靜態的,就不能隨意的訪問外部類的成員變量了,此時內部類只能訪問到外部類的靜態成員變數。
接下來,分別介紹一下這幾種內部類。
非靜態內部類
當一個類作為另一個類的非靜態成員時,則這個類就是一個非靜態內部類。
建立非靜態內部類的示例程式碼如下:
class OutClass {
class InnerClass {}
}
當我們用javac去編譯的時候,發現生成了兩個.class檔案:OutClass.class和OutClass$InnerClass.class。如下圖所示:
從外部類的非靜態方法中例項化內部類
在外部類中訪問內部類是很容易的,直接建立內部類物件,然後通過物件例項呼叫類內的方法即可。示例程式碼如下:
public class OutClass {
private static int a = 0;
public void makeInner() {
InnerClass inClass = new InnerClass();
inClass.seeOuter();
}
public static void main(String[] args) {
OutClass oClass = new OutClass();
oClass.makeInner();
}
class InnerClass {
public void seeOuter() {
System.out.println(a);
a++;
}
}
}
執行結果如下:
0
從外部類的靜態方法中例項化內部類
在外部類中訪問內部類是比較簡單的,可以直接new出內部類物件,但是如果想在外部類的外部使用內部類,接不能直接new內部類名的方式了,而是需要如下方式:
OutClass.InnerClass innerClass = new OutClass().new InnerClass();
也就是說,在外部呼叫非靜態內部類,需要先例項化外部類,然後通過外部類物件再去例項化內部類。示例程式碼如下:
public class OutClass {
private static int a = 0;
public void makeInner() {
InnerClass inClass = new InnerClass();
inClass.seeOuter();
}
public static void main(String[] args) {
OutClass oClass = new OutClass();
oClass.makeInner();
OutClass.InnerClass innerClass = new OutClass().new InnerClass();
innerClass.seeOuter();
}
class InnerClass {
public void seeOuter() {
System.out.println(a);
a++;
}
}
}
執行結果:
0
1
內部類的this引用
普通的類可以使用this引用當前的物件,內部類也是如此。但是假若內部類想引用外部類當前的物件呢?可以使用如下方式:
外部類名.this
示例程式碼如下:
public class OutClass {
private static int a = 0;
public void makeInner() {
InnerClass inClass = new InnerClass();
inClass.seeOuter();
}
public static void main(String[] args) {
OutClass oClass = new OutClass();
oClass.makeInner();
OutClass.InnerClass innerClass = new OutClass().new InnerClass();
innerClass.seeOuter();
}
class InnerClass {
public void seeOuter() {
System.out.println(this);
System.out.println(OutClass.this);
}
}
}
靜態內部類
上面介紹了非靜態內部類,接下來我們學習神馬是靜態內部類。
靜態內部類就是在外部類中扮演一個靜態成員的角色,建立靜態內部類和建立非靜態內部類的形式很相似,只是class前面多了一個static修飾符。
注意,外部類是不可能使用static修飾符進行修飾的。
示例程式碼如下:
class OutClass {
static class InnerClass {
}
}
用javac命令編譯一下,可以看到一樣都是有兩個.class檔案,如下圖所示:
從外部類的非靜態方法中例項化靜態內部類
從外部類中訪問靜態內部類,和在外部類中訪問非靜態內部類是一樣的。但是,需要注意一點,此時靜態內部類只能訪問外部類的靜態成員,無法訪問非靜態成員了。
示例程式碼如下:
public class OutClass {
private static int a = 0;
private int b = 1;
public void makeInner() {
InnerClass inClass = new InnerClass();
inClass.seeOuter();
}
public static void main(String[] args) {
OutClass oClass = new OutClass();
oClass.makeInner();
}
static class InnerClass {
public void seeOuter() {
System.out.println(this);
System.out.println(a);
// System.out.println(b);
}
}
}
執行結果如下:
OutClass$InnerClass@79a340
0
從外部類靜態方法中例項化靜態內部類
注意:
因為靜態內部類是外部類的靜態成員,而靜態成員是跟類繫結,而不是跟類例項化的物件繫結。所以,在外部類的靜態方法中例項化內部類,是不需要先例項化外部類的。
示例程式碼如下:
public class OutClass {
private static int a = 0;
private int b = 1;
public void makeInner() {
InnerClass inClass = new InnerClass();
inClass.seeOuter();
}
public static void main(String[] args) {
OutClass oClass = new OutClass();
oClass.makeInner();
OutClass.InnerClass inClass = new OutClass.InnerClass();
inClass.seeOuter();
}
static class InnerClass {
public void seeOuter() {
System.out.println(this);
System.out.println(a);
// System.out.println(b);
}
}
}
匿名內部類
匿名內部類在Android應用開發中簡直是氾濫,各種listener物件的實現很多都是通過匿名內部類。
匿名內部類從名字上就可以知道這是代表沒有類名的內部類,通常用來簡化程式碼。
相信寫Java的同學都使用過執行緒,那Thread的時候我們可以傳一個Runnable物件,也可以傳一個匿名內部類。示例程式碼如下:
public class OutClass {
public void testAnonymousClass() {
Thread t = new Thread(new Runnable() {
@Override
public void run() {
for (int i = 0; i < 10; i ++) {
System.out.println(i);
try {
Thread.sleep(500);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
});
t.start();
System.out.println("another thread is running...");
}
public static void main(String[] args) {
OutClass oClass = new OutClass();
oClass.testAnonymousClass();
}
}
執行結果如下:
another thread is running...
0
1
2
3
4
5
6
7
8
9
方法內部類
方法內部類顧名思義,就是將類放在方法中,然後它的作用域也是方法的作用域。
然而,我認為方法內部類並沒有什麼卵用,所以這裡不講了,就是這麼任性。