轉自:http://www.cnblogs.com/nerxious/archive/2013/01/24/2875649.html
和:http://www.cnblogs.com/nerxious/archive/2013/01/25/2876489.html 的內容和評論區
內部類不是很好理解,但說白了其實也就是一個類中還包含着另外一個類
如同一個人是由大腦、肢體、器官等身體結果組成,而內部類相當於其中的某個器官之一,例如心臟:它也有自己的屬性和行爲(血液、跳動)
顯然,此處不能單方面用屬性或者方法表示一個心臟,而需要一個類
而心臟又在人體當中,正如同是內部類在外部內當中
實例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$In.class
其中$代表了上面程序中Out.In中的那個 .
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在這裏並沒有特殊含義,只是一種表示形式而已
因爲In這個內部類被私有了嘛,所以無法通過常規的方式生成內部類對象,只能生成Out類的對象去控制內部類(也就是說,此時只有在Out內部才能實例化或操作In的對象),你知道private權限本身就是僅限此類當中啊。這也是爲什麼普通類不會加private修飾的原因,連看都看不到其內部是啥模樣,更別說生成對象了。不信你可以試試上面的代碼,看能否編譯成功。
總之,內部類挺糾結的,它能隨意訪問外部類屬性,也能隨意被包含它的外部類訪問。
內部類的優點是:內部類可以訪問外部類的私有成員變量,而不需要new外部類的對象。
內部類又分爲:靜態內部類、匿名內部類、局部內部類、成員內部類。
靜態內部類的應用場景是:只可以訪問外部類的靜態成員變量和靜態成員方法。
成員內部類的應用場景是:它可以訪問它的外部類的所有成員變量和方法,不管是靜態的還是非靜態的都可以。
局部內部類:像局部變量一樣,不能被public, protected, private和static修飾。只能訪問方法中定義的final類型的局部變量。
匿名內部類:匿名內部類就是沒有名字的局部內部類,不使用關鍵字class, extends, implements, 沒有構造方法。匿名內部類隱式地繼承了一個父類或者實現了一個接口。匿名內部類使用得比較多,通常是作爲一個方法參數。
java 匿名內部類
匿名內部類也就是沒有名字的內部類
正因爲沒有名字,所以匿名內部類只能使用一次,它通常用來簡化代碼編寫
但使用匿名內部類還有個前提條件:必須繼承一個父類或實現一個接口
實例1:不使用匿名內部類來實現抽象方法
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
|
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類只使用一次,那麼將其編寫爲獨立的一個類豈不是很麻煩?
這個時候就引入了匿名內部類
實例2:匿名內部類的基本實現
1
2
3
4
5
6
7
8
9
10
11
12
13
14
|
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中的方法在大括號中實現了
這樣便可以省略一個類的書寫
並且,匿名內部類還能用於接口上
實例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
由上面的例子可以看出,只要一個類是抽象的或是一個接口,那麼其子類中的方法都可以使用匿名內部類來實現
最常用的情況就是在多線程的實現上,因爲要實現多線程必須繼承Thread類或是繼承Runnable接口
實例4:Thread類的匿名內部類實現
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(); } } |
運行結果:1 2 3 4 5
實例5:Runnable接口的匿名內部類實現
1
2
3
4
5
6
7
8
9
10
11
12
13
|
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(); } } |
運行結果:1 2 3 4 5
終極版本
public
class
Demo6{
public
static
void
main(String[]
args) {
new
Thread(
new
Runnable()
{
public
void
run()
{
for
(
int
i
=
1
;
i <=
5
;
i++) {
System.out.print(i
+
"
"
);
}
}
}).start();
}
}