Java基礎FAQ
Java基礎 FAQ
三、I/O篇
18 我怎麼給java程序加啟動參數,就像dir /p/w那樣?
答:還記得public static void main(String[] args)嗎?這裡的args就是你的啟動參數。
在運行時你輸入java package1.class1 -arg1 -arg2,args中就會有兩個String,一個是arg1,另一個是arg2。
19 我怎麼從鍵盤輸入一個int/double/字符串?
答:java的I/O操作比C++要復雜一點。如果要從鍵盤輸入,樣例代碼如下:
BufferedReader cin = new BufferedReader( new InputStreamReader( System.in ) )
;
String s = cin.readLine();
這樣你就獲得了一個字符串,如果你需要數字的話再加上:
int n = Integer.parseInt( s );
或者
double d = Double.parseDouble( s );
20 我怎麼輸出一個int/double/字符串?
答:在程序開始寫:
PrintWriter cout = new PrintWriter( System.out );
需要時寫:
cout.print(n);
或者
cout.println("hello")
等等。
21 我發現有些書上直接用System.in和System.out輸入輸出,比你要簡單得多。
答:java使用unicode,是雙字節。而System.in和System.out是單字節的stream。如果你要輸入輸出雙字節文字比如中文,請使用作者的做法。
四、 關鍵字篇
25 java裡面怎麼定義宏?
答:java不支持宏,因為宏代換不能保證類型安全。如果你需要定義常量,可以將它定義為某個類的static final成員。參見26和30。
26 java裡面沒法用const。
答:你可以用final關鍵字。例如 final int m = 9。被聲明為final的變量不能被再次賦值。也可以用於聲明方法或類,被聲明為final的方法或類不能被繼承。注意const是java的保留字以備擴充。
27 java裡面也不能用goto。
答:甚至在面向過程的語言中你也可以完全不用goto。請檢查你的程序流程是否合理。如果你需要從多層循環中迅速跳出,java增強了(和C++相比)break和continue的功能。
例如:
outer :
while( ... )
{
inner :
for( ... )
{
... break inner; ...
... continue outer; ...
}
}
和const一樣,goto也是java的保留字以備擴充。
28 java裡面能不能重載操作符?
答:不能。String的+號是唯一一個內置的重載操作符。你可以通過定義接口和方法來實現類似功能。
29 我new了一個對象,但是沒法delete掉它。
答:java有自動內存回收機制,即所謂Garbarge Collector。你再也不用擔心指針錯誤。
30 我想知道為什麼main方法必須被聲明為public static?
答:聲明為public是為了這個方法可以被外部調用,詳情見面向對象篇37。
static是為了將某個成員變量/方法關聯到類(class)而非實例(instance)。你不需要創建一個對象就可以直接使用這個類的static成員,在A類中調用B類的static成員可以使用B.staticMember的寫法。注意一個類的static成員變量是唯一的,被所有該類對象所共享的。
31 throw和throws有什麼不同?
答:throws用於聲明一個方法會拋出哪些異常。而throw是在方法體中實際執行拋出異常的動作。如果你在方法中throw一個異常,卻沒有在方法聲明中聲明之,編譯器會報錯。注意Error和RuntimeException的子類是例外,無需特別聲明。
32 什麼是異常?
答:異常最早在Ada語言中引入,用於在程序中動態處理錯誤並恢復。你可以在方法中攔截底層異常並處理之,也可以拋給更高層的模塊去處理。你也可以拋出自己的異常指示發生了某些不正常情況。常見的攔截處理代碼如下:
try
{
...... //以下是可能發生異常的代碼
...... //異常被拋出,執行流程中斷並轉向攔截代碼。
......
}
catch(Exception1 e) //如果Exception1是Exception2的子類並要做特別處理,應排在前面
{
//發生Exception1時被該段攔截
}
catch(Exception2 e)
{
//發生Exception2時被該段攔截
}
finally //這是可選的
{
//無論異常是否發生,均執行此段代碼
}
33 final和finally有什麼不同?
答:final請見26。finally用於異常機制,參見32。
五、 面向對象篇34 extends和implements有什麼不同?
答:extends用於(單)繼承一個類(class),而implements用於實現一個接口(interface)。interface的引入是為了部分地提供多繼承的功能。
在interface中只需聲明方法頭,而將方法體留給實現的class來做。這些實現的class的實例完全可以當作interface的實例來對待。有趣的是在interface之間也可以聲明為extends(單繼承)的關系。
35 java怎麼實現多繼承?
答:java不支持顯式的多繼承。因為在顯式多繼承的語言例如c++中,會出現子類被迫聲明祖先虛基類構造函數的問題,而這是違反面向對象的封裝性原則的。java提供了interface和implements關鍵字來部分地實現多繼承。參見34。
36 abstract是什麼?
答:被聲明為abstract的方法無需給出方法體,留給子類來實現。而如果一個類中有abstract方法,那麼這個類也必須聲明為abstract。被聲明為abstract的類無法實例化,盡管它可以定義構造方法供子類使用。
37 public,protected,private有什麼不同?
答:這些關鍵字用於聲明類和成員的可見性。
public成員可以被任何類訪問,
protected成員限於自己和子類訪問,
private成員限於自己訪問。
Java還提供了第四種的默認可見性,一般稱為package private,當沒有任何public,protected,private修飾符時,成員是同一包內可見。類可以用public或默認來修飾。
38 Override和Overload有什麼不同?
答:Override是指父類和子類之間方法的繼承關系,這些方法有著相同的名稱和參數類型。Overload是指同一個類中不同方法(可以在子類也可以在父類中定義)間的關系,這些方法有著相同的名稱和不同的參數類型。
39 我繼承了一個方法,但現在我想調用在父類中定義的方法。
答:用super.xxx()可以在子類中調用父類方法。
40 我想在子類的構造方法中調用父類的構造方法,該怎麼辦?
答:在子類構造方法的第一行調用super(...)即可。
41 我在同一個類中定義了好幾個構造方法並且想在一個構造方法中調用另一個。
答:在構造方法第一行調用this(...)。
42 我沒有定義構造方法會怎麼樣?
答:自動獲得一個無參數的構造方法。
43 我調用無參數的構造方法失敗了。
答:如果你至少定義了一個構造方法,就不再有自動提供的無參數的構造方法了。你需要顯式定義一個無參數的構造方法。
44 我該怎麼定義類似於C++中的析構方法(destructor)?
答:提供一個void finalize()方法。在Garbarge Collector回收該對象時會調用該方法。注意實際上你很難判斷一個對象會在什麼時候被回收。作者從未感到需要提供該方法。
45 我想將一個父類對象轉換成一個子類對象該怎麼做?
答:強制類型轉換。如
public void meth(A a)
{
B b = (B)a;
}
如果a實際上並不是B的實例,會拋出ClassCastException。所以請確保a確實是B的實例。
46 其實我不確定a是不是B的實例,能不能分情況處理?
答:可以使用instanceof操作符。例如
if( a instanceof B )
{
B b = (B)a;
}
else
{
...
}
47 我在方法裡修改了一個對象的值,但是退出方法後我發現這個對象的值沒變!
答:很可能你把傳入參數重賦了一個新對象,例如下列代碼就會造成這種錯誤:
public void fun1(A a) //a是局部參數,指向了一個外在對象。
{
a = new A(); //a指向了一個新對象,和外在對象脫鉤了。如果你要讓a作為傳出變量,不要寫這一句。
a.setAttr(attr);//修改了新對象的值,外在對象沒有被修改。
}
基本類型也會出現這種情況。例如:
public void fun2(int a)
{
a = 10;//只作用於本方法,外面的變量不會變化。
}
六、java.util篇