內部類的使用:
內部類的使用的優缺點:
?? 優點:減少了在類文件編譯后的產生的字節碼文件的大小
?? 缺點:使程序結構不清晰
使用內部類的注意事項:
?? 內部類的使用一般都與所在的外部類有一定的關聯,它是在一個類的內部嵌套定義的類,它可以是其它類的成員,也可以在一個語句塊的內部定義,還可以在表達式內部匿名定義(匿名內部類)。
內部類有如下的特性:
? .一般用在定義它的類或語句塊之內,在外部引用它時必須給出完整的名稱,名字不能與包含它的數百名相同
?. 可以使用包含它的類的表態和實例成員變量,也可以使用它所在方法的局部變量
?.可以定義為abstract
?.若被聲明為static,就變成了頂層類,不能再使用局部變量
?.若想在inner class 中聲明任何static成員,則該inner class必須聲明為static.
?.匿名類是一種特殊的內部類,它是在一個表達式的內部包含一個完整的類的定義。匿名內部類不需要任何的修飾詞。
?.內部類可以使用任何聽修飾符,但是如果聲明為static類,則等同于一個標準類。
?.如果非表態內部類,就擁有對外部類的所有成員的完全訪問權限,包括實例字段和方法。為實現這一行為,非表態內部類存著對外部類的實例的一個隱匿的引用。
?.所以對一個非表態內部類進行實例化需要采用不同語法的new 語句,這種開工的new 語句要求外部類的一個實例,使內部類能在那個實例的上下文中創建。
?.非表態內部類具有一些限制。尤其是,它們不能聲明表態初始化列表和表態成員,除非是在常量字段中。此外方法內部聲明的內部類不能訪問方法的局部變量和參數,除非它們被初始化成final.
?.表態內部類當中可以有表態數據,表態方法或者是又一個表態內部類。而非表態內部類當中不能有靜態數據。這是它們的區別。
?.局部內部類:Java內部類也可以是局部的,它可以定義在一個方法甚至一個代碼塊之內。
?
有一點需要注意的是,匿名內部類由于沒有名字,所以它沒有構造函數(但是如果這個匿名內部類繼承了一個只含有帶參數構造函數的父類,創建它的時候必須帶上這些參數,并在實現的過程中使用super關鍵字調用相應的內容)。如果你想要初始化它的成員變量,有下面幾種方法:
如果是在一個方法的匿名內部類,可以利用這個方法傳進你想要的參數,不過記住,這些參數必須被聲明為final。
將匿名內部類改造成有名字的局部內部類,這樣它就可以擁有構造函數了。
在這個匿名內部類中使用初始化代碼塊。
為什么需要內部類?
java內部類有什么好處?為什么需要內部類?
首先舉一個簡單的例子,如果你想實現一個接口,但是這個接口中的一個方法和你構想的這個類中的一個方法的名稱,參數相同,你應該怎么辦?這時候,你可以建一個內部類實現這個接口。由于內部類對外部類的所有內容都是可訪問的,所以這樣做可以完成所有你直接實現這個接口的功能。
不過你可能要質疑,更改一下方法的不就行了嗎?
的確,以此作為設計內部類的理由,實在沒有說服力。
真正的原因是這樣的,java中的內部類和接口加在一起,可以的解決常被C++程序員抱怨java中存在的一個問題——沒有多繼承。實際上,C++的多繼承設計起來很復雜,而java通過內部類加上接口,可以很好的實現多繼承的效果。
下面是使用內部類的幾個小例子:
?JAVA的內部類的構造方法是不會自動調用,必須在外部類當中的某個方法當中或構造函數當中顯示調用才能夠正確使用。
?? Class Outer{
??
??????? public outer(){
?System.out.println("....");
?????? }
??????? Class Inner{
?public Innter(){
?
?????? System.out.println("inner");
?}
??????? }
?????? public static void Main(String[] args){
??????????? Outer a = new Outer();
?????? }
}
這種使用方法不會關系到內部類的輸出。因為它并沒有顯示調用內部類。
所以我們應該進行這樣的修改:
? 在Outer構造函數中加入 Innter b = new Innter();
就可以直接調用它的輸出或方法體了。
不過我們經常使用的還是匿名內部類:
?? class Outer{
?????? Outer(){
?(new Inner(){
??public void print(){
???System.out.println("Inner");
???}
??}
?).print();
?System.out.println("Outer");
?????? }
}
?
下面是JAVA的一個源碼使用了匿名內部類:
/**
* "Tokenizes" the entire stream as a single token.
*/
public class KeywordAnalyzer extends Analyzer {
public TokenStream tokenStream(String fieldName,
final Reader reader) {
return new TokenStream() {
private boolean done;
private final char[] buffer = new char[1024];
public Token next() throws IOException {
if (!done) {
done = true;
StringBuffer buffer = new StringBuffer();
int length = 0;
while (true) {
length = reader.read(this.buffer);
if (length == -1) break;
buffer.append(this.buffer, 0, length);
}
String text = buffer.toString();
return new Token(text, 0, text.length());
}
return null;
}
};
}
}
posted on 2006-08-01 08:53
水煮三國 閱讀(7282)
評論(3) 編輯 收藏 所屬分類:
J2SE