31 throw和throws有什么不同?
答:throws用于聲明一個(gè)方法會(huì)拋出哪些異常。而throw是在方法體中實(shí)際執(zhí)行拋出異常的動(dòng)作。
如果你在方法中throw一個(gè)異常,卻沒有在方法聲明中聲明之,編譯器會(huì)報(bào)錯(cuò)。
注意Error和RuntimeException的子類是例外,無需特別聲明。
32 什么是異常?
答:異常最早在Ada語言中引入,用于在程序中動(dòng)態(tài)處理錯(cuò)誤并恢復(fù)。
你可以在方法中攔截底層異常并處理之,也可以拋給更高層的模塊去處理。
你也可以拋出自己的異常指示發(fā)生了某些不正常情況。常見的攔截處理代碼如下:
try
{
...... //以下是可能發(fā)生異常的代碼
...... //異常被拋出,執(zhí)行流程中斷并轉(zhuǎn)向攔截代碼。
......
}
catch(Exception1 e) //如果Exception1是Exception2的子類并要做特別處理,應(yīng)排在前面
{
//發(fā)生Exception1時(shí)被該段攔截
}
catch(Exception2 e)
{
//發(fā)生Exception2時(shí)被該段攔截
}
finally //這是可選的
{
//無論異常是否發(fā)生,均執(zhí)行此段代碼
}
33 final和finally有什么不同?
答:final請(qǐng)見26。finally用于異常機(jī)制,參見32。
五、 面向?qū)ο笃?/b>
34 extends和implements有什么不同?
答:extends用于(單)繼承一個(gè)類(class),而implements用于實(shí)現(xiàn)一個(gè)接口(interface)。
interface的引入是為了部分地提供多繼承的功能。
在interface中只需聲明方法頭,而將方法體留給實(shí)現(xiàn)的class來做。
這些實(shí)現(xiàn)的class的實(shí)例完全可以當(dāng)作interface的實(shí)例來對(duì)待。
有趣的是在interface之間也可以聲明為extends(單繼承)的關(guān)系。
35 java怎么實(shí)現(xiàn)多繼承?
答:java不支持顯式的多繼承。
因?yàn)樵陲@式多繼承的語言例如c++中,會(huì)出現(xiàn)子類被迫聲明祖先虛基類構(gòu)造函數(shù)的問題,而這是違反面向?qū)ο蟮姆庋b性原則的。
java提供了interface和implements關(guān)鍵字來部分地實(shí)現(xiàn)多繼承。參見34。
36 abstract是什么?
答:被聲明為abstract的方法無需給出方法體,留給子類來實(shí)現(xiàn)。
而如果一個(gè)類中有abstract方法,那么這個(gè)類也必須聲明為abstract。
被聲明為abstract的類無法實(shí)例化,盡管它可以定義構(gòu)造方法供子類使用。
37 public,protected,private有什么不同?
答:這些關(guān)鍵字用于聲明類和成員的可見性。public成員可以被任何類訪問,protected成員限于自己和子類訪問,private成員限于自己訪問。
Java還提供了第四種的默認(rèn)可見性,當(dāng)沒有任何public,protected,private修飾時(shí).
類可以用public或默認(rèn)來修飾。
38 Override和Overload有什么不同?
答:Override是指父類和子類之間方法的繼承關(guān)系,這些方法有著相同的名稱和參數(shù)類型。
Overload是指同一個(gè)類中不同方法(可以在子類也可以在父類中定義)間的關(guān)系,這些方法有著相同的名稱和不同的參數(shù)類型。
39 我繼承了一個(gè)方法,但現(xiàn)在我想調(diào)用在父類中定義的方法。
答:用super.xxx()可以在子類中調(diào)用父類方法。
40 我想在子類的構(gòu)造方法中調(diào)用父類的構(gòu)造方法,該怎么辦?
答:在子類構(gòu)造方法的第一行調(diào)用super(...)即可。
41 我在同一個(gè)類中定義了好幾個(gè)構(gòu)造方法并且想在一個(gè)構(gòu)造方法中調(diào)用另一個(gè)。
答:在構(gòu)造方法第一行調(diào)用this(...)。
42 我沒有定義構(gòu)造方法會(huì)怎么樣?
答:自動(dòng)獲得一個(gè)無參數(shù)的構(gòu)造方法。
43 我調(diào)用無參數(shù)的構(gòu)造方法失敗了。
答:如果你至少定義了一個(gè)構(gòu)造方法,就不再有自動(dòng)提供的無參數(shù)的構(gòu)造方法了。
你需要顯式定義一個(gè)無參數(shù)的構(gòu)造方法。
44 我該怎么定義類似于C++中的析構(gòu)方法(destructor)?
答:提供一個(gè)void finalize()方法。在Garbarge Collector回收該對(duì)象時(shí)會(huì)調(diào)用該方法。
注意實(shí)際上你很難判斷一個(gè)對(duì)象會(huì)在什么時(shí)候被回收。作者從未感到需要提供該方法。
45 我想將一個(gè)父類對(duì)象轉(zhuǎn)換成一個(gè)子類對(duì)象該怎么做?
答:強(qiáng)制類型轉(zhuǎn)換。如
public void meth(A a)
{
B b = (B)a;
}
如果a實(shí)際上并不是B的實(shí)例,會(huì)拋出ClassCastException。所以請(qǐng)確保a確實(shí)是B的實(shí)例。
46 其實(shí)我不確定a是不是B的實(shí)例,能不能分情況處理?
答:可以使用instanceof操作符。例如
if( a instanceof B )
{
B b = (B)a;
}
else
{
...
}
47 我在方法里修改了一個(gè)對(duì)象的值,但是退出方法后我發(fā)現(xiàn)這個(gè)對(duì)象的值沒變!
答:很可能你把傳入?yún)?shù)重賦了一個(gè)新對(duì)象,例如下列代碼就會(huì)造成這種錯(cuò)誤:
public void fun1(A a) //a是局部參數(shù),指向了一個(gè)外在對(duì)象。
{
a = new A(); //a指向了一個(gè)新對(duì)象,和外在對(duì)象脫鉤了。如果你要讓a作為傳出變量,不要寫這一句。
a.setAttr(attr);//修改了新對(duì)象的值,外在對(duì)象沒有被修改。
}
基本類型也會(huì)出現(xiàn)這種情況。例如:
public void fun2(int a)
{
a = 10;//只作用于本方法,外面的變量不會(huì)變化。
}
六、java.util篇
48 java能動(dòng)態(tài)分配數(shù)組嗎?
答:可以。例如int n = 3; Language[] myLanguages = new Language[n];
49 我怎么知道數(shù)組的長(zhǎng)度?
答:用length屬性。如上例中的 myLanguages.length 就為 3。
50 我還想讓數(shù)組的長(zhǎng)度能自動(dòng)改變,能夠增加/刪除元素。
答:用順序表--java.util.List接口。
你可以選擇用ArrayList或是LinkedList,前者是數(shù)組實(shí)現(xiàn),后者是鏈表實(shí)現(xiàn)。
例如: List list = new ArrayList(); 或是 List list = new LinkedList(); 。
51 什么是鏈表?為什么要有兩種實(shí)現(xiàn)?
答:請(qǐng)補(bǔ)習(xí)數(shù)據(jù)結(jié)構(gòu)。
52 我想用隊(duì)列/棧。
答:用java.util.LinkedList。
53 我希望不要有重復(fù)的元素。
答:用集合--java.util.Set接口。例如:Set set = new HashSet()。
54 我想遍歷集合/Map。
答:用java.util.Iterator。參見API。
55 我還要能夠排序。
答:用java.util.TreeSet。例如:Set set = new TreeSet()。放進(jìn)去的元素會(huì)自動(dòng)排序。
你需要為元素實(shí)現(xiàn)Comparable接口,還可能需要提供equals()方法,compareTo()方法,hash Code()方法。
56 但是我想給數(shù)組排序。
答:java.util.Arrays類包含了sort等實(shí)用方法。
57 我想按不同方法排序。
答:為每種方法定義一個(gè)實(shí)現(xiàn)了接口Comparator的類并和Arrays綜合運(yùn)用。
58 Map有什么用?
答:存儲(chǔ)key-value的關(guān)鍵字-值對(duì),你可以通過關(guān)鍵字來快速存取相應(yīng)的值。
59 set方法沒問題,但是get方法返回的是Object。
答:強(qiáng)制類型轉(zhuǎn)換成你需要的類型。參見45。
60 我要獲得一個(gè)隨機(jī)數(shù)。
答:使用java.util.Random類。
61 我比較兩個(gè)String總是false,但是它們明明都是"abc" !
答:比較String一定要使用equals或equalsIgnoreCase方法,不要使用 == !
==比較的是兩個(gè)引用(變量)是否指向了同一個(gè)對(duì)象,而不是比較其內(nèi)容。