六個例項教你正確使用Java內部類(成員內部類、匿名內部類、靜態內部類、區域性內部類)
內部類不是很好理解,但說白了其實也就是一個類中還包含著另外一個類
如同一個人是由大腦、肢體、器官等身體結果組成,而內部類相當於其中的某個器官之一,例如心臟:它也有自己的屬性和行為(血液、跳動)
顯然,此處不能單方面用屬性或者方法表示一個心臟,而需要一個類
而心臟又在人體當中,正如同是內部類在外部內當中
例項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.class和Out
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.屬性名
例項3:靜態內部類
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();
}
}
執行結果:12
可以看到,如果用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();
}
}
執行結果:12
如果一個內部類只希望被外部類中的方法操作,那麼可以使用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);
}
}
執行結果:
3
12
在上面的程式碼中,我們將內部類移到了外部類的方法中,然後在外部類的方法中再生成一個內部類物件去呼叫內部類方法
如果此時我們需要往外部類的方法中傳入引數,那麼外部類的方法形參必須使用final定義
至於final在這裡並沒有特殊含義,只是一種表示形式而已
匿名內部類也就是沒有名字的內部類
正因為沒有名字,所以匿名內部類只能使用一次,它通常用來簡化程式碼編寫
但使用匿名內部類還有個前提條件:必須繼承一個父類或實現一個介面
例項6.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類只使用一次,那麼將其編寫為獨立的一個類豈不是很麻煩?
這個時候就引入了匿名內部類
例項6.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中的方法在大括號中實現了
這樣便可以省略一個類的書寫
並且,匿名內部類還能用於介面上
例項6.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();
}
}
執行結果:eat something
由上面的例子可以看出,只要一個類是抽象的或是一個介面,那麼其子類中的方法都可以使用匿名內部類來實現