Java學習入門篇.必看.
1什麼是Java、Java2、JDK?JDK後面的1.3、1.4.2版本號又是怎麼回事?
答:Java是一種通用的,並發的,強類型的,面向對象的編程語言(摘自Java規范第二版)JDK是Sun公司分發的免費Java開發工具,正式名稱為J2SDK(Java2Softw
areDevelopKit)。
2什麼是JRE/J2RE?
答:J2RE是Java2RuntimeEnvironment,即Java運行環境,有時簡稱JRE。
如果你只需要運行Java程序或Applet,下載並安裝它即可。
如果你要自行開發Java軟件,請下載JDK。在JDK中附帶有J2RE。
注意由於Microsoft對Java的支持不完全,請不要使用IE自帶的虛擬機來運行Applet,務
必安
裝一個J2RE或JDK。
3學習Java用什麼工具比較好?
答:作者建議首先使用JDK+文本編輯器,這有助你理解下列幾個基礎概念:path,class
path,package
並熟悉基本命令:javac和Java。並且下載和你的JDK版本一致的API幫助。
如果你不確定類或函數的用法,請先查閱API而不是發貼求助。
當你熟悉Java之後,你可以考慮換一個IDE。很多人推薦JCreator,實際上JCreator的功能還是很弱的。
作者推薦eclipse,下載網址http://www.eclipse.org;。因eclispe是免費的.
4學習Java有哪些好的參考書?
答:作者首先推薦ThinkinginJava,中文名《Java編程思想》,有中文版。
該書第一章介紹了很多面向對象的編程思想,作為新手應當認真閱讀。
除此以外,O′relly出版社和Wrox出版社的書也不錯。作者本人不喜歡大陸作者的書。
也許你覺得英文太難,但是網上大多數資料都是英文的。另外,你需要經常查閱API,而
那也是英文的。
5Java和C++哪個更好?
答:這個問題是一個很不恰當的問題。你應該問:Java和C++哪個更適用於我的項目?
如果你不需要跨平台,不需要分布式,要強調程序的運行速度,C++更為適用。
反之?你應當考慮Java。
6什麼是J2SE/J2EE/J2ME?
答:J2SE就是一般的Java。
J2ME是針對嵌入式設備的,比如Java手機,它有自己的SDK。而J2EE使用J2SE的SDK。
J2EE規范更多的是對J2EE服務器的要求和開發人員的約束。詳情見後繼《J2EEFAQ》。
二、命令篇
7我寫了第一個Java程序,應該如何編譯/運行?
答:首先請將程序保存為xxx.Java文件,
然後在DOS窗口下使用javacxxx.Java命令,你會發現該目錄下多了一個xxx.class文件,
再使用javaxxx命令,你的Java程序就開始運行了。
8我照你說的做了,但是出現什麼"′Javac′不是內部或外部命令,也不是可運行的程序或批處理文件。"。
答:你遇到了path問題。操作系統在一定的范圍(path)內搜索Javac.exe,但是沒能找到。
請編輯你的操作系統環境變量,新增一個Java_HOME變量,設為你JDK的安裝目錄,
再編輯Path變量,加上一項%Java_HOME%in。
然後關掉並新開一個DOS窗口,你就可以使用javac和Java命令了。
9環境變量怎麼設置?
答:請向身邊會設的人咨詢。
10javacxxx.java順利通過了,但是Javaxxx的時候顯示什麼"NoClassDefFoundError"?br>
?br>答:你遇到了classpath問題。Java命令在一定的范圍(classpath)內搜索你要用的
class文件,但是未能找到。
首先請確認你沒有錯敲成Javaxxx.class,
其次,檢查你的CLASSPATH環境變量,如果你設置了該變量又沒有包含.(代表當前目錄
)的?br>睿?br>你就會遇到這個問題。請在你的CLASSPATH環境變量中加入一項.。另
外參見15。
11我在javaxxx的時候顯示"Exceptioninthread"main"Java.lang.NoSuchMethod
Error:main"。
答:首先,在你的程序中每個Java文件有且只能有一個public類,
這個類的類名必須和文件名的大小寫完全一樣。
其次,在你要運行的類中有且只能有一個publicstaticvoidmain(String[]args)方法,
這個方法就是你的主程序。
12package是什麼意思?怎麼用?
答:為了唯一標識每個類並分組,Java使用了package的概念。
每個類都有一個全名,例如String的全名是java.lang.String,其中Java.lang是包名,
String是短名。
這樣,如果你也定義了String,你可以把它放在mypackage中,
通過使用全名mypackage.String和Java.lang.String來區分這兩個類。
同時,將邏輯上相關的類放在同一個包中,可以使程序結構更為清楚。
你要做的就是在Java文件開頭加一行"packagemypackage;"。
注意包沒有嵌套或包含關系,A包和A.B包對Java命令來說是並列的兩個包(雖然開發者
可能br>凳景?含關系)br>
13我沒有聲明任何package會怎麼樣?
答:你的類被認為放在默認包中。這時全名和短名是一致的。
14在一個類中怎麼使用其他類?
答:如果你使用Java.lang包中的類,不用做任何事。
如果你使用其他包中的類,使用importpackage1.class1;或importpackage2.*;
這裡.*表示引入這個包中的所有類。然後在程序中你可以使用其他類的短名。
如果短名有沖突,使用全名來區分。
15我用了package的時候顯示"NoClassDefFoundError",但是我把所有package去掉的時
候能正常運行。
答:將你的Java文件按包名存放。
比如你的工作目錄是/work,你的類是package1.class1,那麼將它存放為/work/package
1/class1.Java。
如果沒有聲明包,那麼直接放在/work下。
在/work下執行javacpackage1/class1.java,再執行Javapackage1.class1,你會發現
一切正常。
另外,你可以考慮開始使用IDE。
16我想把Java編譯成exe文件,該怎麼做?
答:JDK只能將Java源文件編譯為class文件。
class文件是一種跨平台的字節碼,必須依賴平台相關的JRE來運行。Java以此來實現跨
平台?br>浴?br>有些開發工具可以將Java文件編譯為exe文件。作者反對這種做法,因
為這樣就取消了跨平台性。
如果你確信你的軟件只在Windows平台上運行,你可以考慮使用C++/C#來編程。
17我在編譯的時候遇到什麼"deprecatedAPI",是什麼意思?
答:所謂deprecated是指已經?時,但是為了向前兼容起見仍然保留的方法??br>這些方
法可能會在以後取消支持。你應當改用較新的方法。
一般在API裡面會說明你應當用什麼方法來代替之。
三、I/O篇
18我怎麼給Java程序加啟動參數,就像dir/p/w那樣?
答:還記得publicstaticvoidmain(String[]args)嗎?這裡的args就是你的啟動參數。
在運行時你輸入Javapackage1.class1-arg1-arg2,args中就會有兩個String,一個
是arg1,另一個是arg2。
19我怎麼從鍵盤輸入一個int/double/字符串?
答:Java的I/O操作比C++要復雜一點。如果要從鍵盤輸入,樣例代碼如下:
BufferedReadercin=newBufferedReader(newInputStreamReader(System.in));
Strings=cin.readLine();
這樣你就獲得了一個字符串,如果你需要數字的話再加上:
intn=Integer.parseInt(s);或者doubled=Double.parseDouble(s);
20我怎麼輸出一個int/double/字符串?
答:在程序開始寫:
PrintWritercout=newPrintWriter(System.out);
需要時寫:
cout.print(n);或者cout.println("hello")等等。
21我發現有些書上直接用System.in和System.out輸入輸出,比你要簡單得多。
答:Java使用unicode,是雙字節。而System.in和System.out是單字節的stream。
如果你要輸入輸出雙字節文字比如中文,請使用作者的做法。
22我怎麼從文件輸入一個int/double/字符串?
答:類似於從鍵盤輸入,只不過換成
BufferedReaderfin=newBufferedReader(newFileReader("myFileName"));
PrintWriterfout=newPrintWriter(newFileWriter("myFileName"));
另外如果你還沒下載API,請開始下載並閱讀Java.io包中的內容。
23我想讀寫文件的指定位置,該怎麼辦?
答:你肯定沒有認真看API。Java.io.RandomAccessFile可以滿足你的需要。
24怎麼判斷要讀的文件已經到了盡頭?
答:你肯定沒有認真看API。在Reaer的read方法中明確說明返回-1表示流的結尾。
四、關鍵字篇
25Java裡面怎麼定義宏?
答:Java不支持宏,因為宏代換不能保證類型安全。
如果你需要定義常量,可以將它定義為某個類的staticfinal成員。參見26和30。
26Java裡面沒法用const。
答:你可以用final關鍵字。例如finalintm=9。被聲明為final的變量不能被再次
賦值?br>?br>final也可以用於聲明方法或類,被聲明為final的方法或類不能被繼承。
注意const是Java的保留字以備擴充。
27Java裡面也不能用goto。
答:甚至在面向過程的語言中你也可以完全不用goto。請檢查你的程序流程是否合理。
如果你需要從多層循環中迅速跳出,Java增強了(和C++相比)break和continue的功能,支?br>?label。
例如:
outer:
while(...)
{
inner:
for(...)
{
...breakinner;...
...continueouter;...
}
}
和const一樣,goto也是Java的保留字以備擴充。
28Java裡面能不能重載操作符?
答:不能。String的+號是唯一一個內置的重載操作符。你可以通過定義接口和方法來實
現類似功能。
29我new了一個對象,但是沒法delete掉它。
答:Java有自動內存回收機制,即所謂GarbargeCollector。你再也不用擔心指針錯誤,內?br>嬉绯隽恕?br>
30我想知道為什麼main方法必須被聲明為publicstatic?
答:聲明為public是為了這個方法可以被外部調用,詳情見面向對象篇37。
static是為了將某個成員變量/方法關聯到類(class)而非實例(instance)。
你不需要創建一個對象就可以直接使用這個類的static成員,
在A類中調用B類的static成員可以使用B.staticMember的寫法。
注意一個類的static成員變量是唯一的,被所有該類對象所共享的。
31throw和throws有什麼不同?
答:throws用於聲明一個方法會拋出哪些異常。而throw是在方法體中實際執行拋出異常的動
作。如果你在方法中throw一個異常,卻沒有在方法聲明中聲明之,編譯器會報錯。
注意Error和RuntimeException的子類是例外,無需特別聲明。
32什麼是異常?
答:異常最早在Ada語言中引入,用於在程序中動態處理錯誤並恢復。
你可以在方法中攔截底層異常並處理之,也可以拋給更高層的模塊去處理。
你也可以拋出自己的異常指示發生了某些不正常情況。常見的攔截處理代碼如下:
try
{
......//以下是可能發生異常的代碼
......//異常被拋出,執行流程中斷並轉向攔截代碼。
......
}
catch(Exception1e)//如果Exception1是Exception2的子類並要做特別處理,應排在
前面
{
//發生Exception1時被該段攔截
}
catch(Exception2e)
{
//發生Exception2時被該段攔截
}
finally//這是可選的
{
//無論異常是否發生,均執行此段代碼
}
33final和finally有什麼不同?
答:final請見26。finally用於異常機制,參見32。
五、面向對象篇
34extends和implements有什麼不同?
答:extends用於(單)繼承一個類(class),而implements用於實現一個接口(inter
face
)。
interface的引入是為了部分地提供多繼承的功能。
在interface中只需聲明方法頭,而將方法體留給實現的class來做。
這些實現的class的實例完全可以當作interface的實例來對待。
有趣的是在interface之間也可以聲明為extends(單繼承)的關系。
35Java怎麼實現多繼承?
答:Java不支持顯式的多繼承。
因為在顯式多繼承的語言例如c++中,會出現子類被迫聲明祖先虛基類構造函數的問題,
而這是違反面向對象的封裝性原則的。
Java提供了interface和implements關鍵字來部分地實現多繼承。參見34。
36abstract是什麼?
答:被聲明為abstract的方法無需給出方法體,留給子類來實現。
而如果一個類中有abstract方法,那麼這個類也必須聲明為abstract。
被聲明為abstract的類無法實例化,盡管它可以定義構造方法供子類使用。
37public,protected,private有什麼不同?
答:這些關鍵字用於聲明類和成員的可見性。
public成員可以被任何類訪問,
protected成員限於自己和子類訪問,
private成員限於自己訪問。
Java還提供了第四種的默認可見性,當沒有任何public,protected,private修飾時,成
員是?br>?一包內可見??br>
類可以用public或默認來修飾。
38Override和Overload有什麼不同?
答:Override是指父類和子類之間方法的繼承關系,這些方法有著相同的名稱和參數類
型。
Overload是指同一個類中不同方法(可以在子類也可以在父類中定義)間的關系,
這些方法有著相同的名稱和不同的參數類型。
39我繼承了一個方法,但現在我想調用在父類中定義的方法。
答:用super.xxx()可以在子類中調用父類方法。
40我想在子類的構造方法中調用父類的構造方法,該怎麼辦?
答:在子類構造方法的第一行調用super(...)即可。
41我在同一個類中定義了好幾個構造方法並且想在一個構造方法中調用另一個。
答:在構造方法第一行調用this(...)。
42我沒有定義構造方法會怎麼樣?
答:自動獲得一個無參數的構造方法。
43我調用無參數的構造方法失敗了。
答:如果你至少定義了一個構造方法,就不再有自動提供的無參數的構造方法了。
你需要顯式定義一個無參數的構造方法。
44我該怎麼定義類似於C++中的析構方法(destructor)?
答:提供一個voidfinalize()方法。在GarbargeCollector回收該對象時會調用該方法。
注意實際上你很難判斷一個對象會在什麼時候被回收。作者從未感到需要提供該方法。
45我想將一個父類對象轉換成一個子類對象該怎麼做?
答:強制類型轉換。如
publicvoidmeth(Aa)
{
Bb=(B)a;
}
如果a實際上並不是B的實例,會拋出ClassCastException。所以請確保a確實是B的實例
。
46其實我不確定a是不是B的實例,能不能分情況處理?
答:可以使用instanceof操作符。例如
if(ainstanceofB)
{
Bb=(B)a;
}
else
{
...
}
47我在方法裡修改了一個對象的值,但是退出方法後我發現這個對象的值沒變!
答:很可能你把傳入參數重賦了一個新對象,例如下列代碼就會造成這種錯誤:
publicvoidfun1(Aa)//a是局部參數,指向了一個外在對象。
{
a=newA();//a指向了一個新對象,和外在對象脫鉤了。如果你要讓a作為傳
出變
量,不要寫這一句。
a.setAttr(attr);//修改了新對象的值,外在對象沒有被修改。
}
基本類型也會出現這種情況。例如:
publicvoidfun2(inta)
{
a=10;//只作用於本方法,外面的變量不會變化。
}
六、Java.util篇
48Java能動態分配數組嗎?
答:可以。例如intn=3;Language[]myLanguages=newLanguage[n];
49我怎麼知道數組的長度?
答:用length屬性。如上例中的myLanguages.length就為3。
50我還想讓數組的長度能自動改變,能夠增加/刪除元素。
答:用順序表--Java.util.List接口。
你可以選擇用ArrayList或是LinkedList,前者是數組實現,後者是鏈表實現。
例如:Listlist=newArrayList();或是Listlist=newLinkedList();。
51什麼是鏈表?為什麼要有兩種實現?
答:請補習數據結構。
52我想用隊列/棧。
答:用Java.util.LinkedList。
53我希望不要有重復的元素。
答:用集合--Java.util.Set接口。例如:Setset=newHashSet()。
54我想遍歷集合/Map。
答:用Java.util.Iterator。參見API。
55我還要能夠排序。
答:用Java.util.TreeSet。例如:Setset=newTreeSet()。放進去的元素會自動排
序。
你需要為元素實現Comparable接口,還可能需要提供equals()方法,compareTo()方法,
hash
Code()方法。
56但是我想給數組排序。
答:Java.util.Arrays類包含了sort等實用方法。
57我想按不同方法排序。
答:為每種方法定義一個實現了接口Comparator的類並和Arrays綜合運用。
58Map有什麼用?
答:存儲key-value的關鍵字-值對,你可以通過關鍵字來快速存取相應的值。
59set方法沒問題,但是get方法返回的是Object。
答:強制類型轉換成你需要的類型。參見45。
60我要獲得一個隨機數。
答:使用Java.util.Random類。
61我比較兩個String總是false,但是它們明明都是"abc"!
答:比較String一定要使用equals或equalsIgnoreCase方法,不要使用==!
==比較的是兩個引用(變量)是否指向了同一個對象,而不是比較其內容。