java入門與加深FAQ(原作者:robbin&cammsia )

一、準備篇

1
什麼是Java、Java2、JDK?JDK後面的1.3、1.4.2版本號又是怎麼回事?
 答:Java是一種通用的,併發的,強類型的,面向對象的編程語言(摘自Java規範第二版) JDK是Sun公司分發的免費Java開發工具,正式名稱爲J2SDK(Java2 Software Develop Kit)。

2 什麼是JRE/J2RE?
 答:J2RE是Java2 Runtime Environment,即Java運行環境,有時簡稱JRE。如果你只需要運行Java程序或Applet,下載並安裝它即可。如果你要自行開發Java軟件,請下載JDK。在JDK中附帶有J2RE。
注意:由於Microsoft對Java的支持不完全,請不要使用IE自帶的虛擬機來運行Applet,務必安裝一個J2RE或JDK。

3 學習Java用什麼工具比較好?
 答:作者建議首先使用JDK+文本編輯器,這有助你理解下列幾個基礎概念:path,classpath,package並熟悉基本命令:javac和java。並且下載和你的JDK版本一致的API幫助。

 如果你不確定類或函數的用法,請先查閱API而不是發貼求助。當你熟悉Java之後,你可以考慮換一個IDE。很多人推薦JCreator,實際上JCreator的功能還 是很弱的。作者推薦eclipse,下載網http://www.eclipse.org因eclispe是免費的.

4 學習Java有哪些好的參考書?
 答:作者首先推薦Thinking in Java,中文名《Java編程思想》,有中文版。該書第一章介紹了很多面向對象的編程思想,作爲新手應當認真閱讀。除此以外,O´relly出版社和Wrox出版社的書也不錯。作者本人不喜歡大陸作者的書。也許你覺得英文太難,但是網上大多數資料都是英文的。另外,你需要經常查閱API,而那也是英文的。

5 Java和C++哪個更好?
 答:這個問題是一個很不恰當的問題。你應該問:Java和C++哪個更適用於我的項目?如果你不需要跨平臺,不需要分佈式,要強調程序的運行速度,C++更爲適用。反之?你應當考慮Java。

6 什麼是J2SE/J2EE/J2ME?
 答:J2SE就是一般的Java。
 J2ME是針對嵌入式設備的,比如Java手機,它有自己的SDK。而J2EE使用J2SE的SDK。
 J2EE規範更多的是對J2EE服務器的要求和開發人員的約束。詳情見後繼"J2EE FAQ"。


二、命令篇

7 我寫了第一個Java程序,應該如何編譯/運行?
 答:首先請將程序保存爲xxx.java文件,然後在dos窗口下使用javac xxx.java命令,你會發現該目錄下多了一個xxx.class文件,再使用java xxx命令,你的java程序就開始運行了。

8 我照你說的做了,但是出現什麼"´javac´ 不是內部或外部命令,也不是可運行的程序或批處理文件。"。
 答:你遇到了path問題。操作系統在一定的範圍(path)內搜索javac.exe,但是沒能找到。請編輯你的操作系統環境變量,新增一個JAVA_HOME變量,設爲你JDK的安裝目錄,再編輯Path變量,加上一項 %JAVA_HOME%/bin。然後關掉並新開一個dos窗口,你就可以使用javac和java命令了。

9 環境變量怎麼設置?
 答:請向身邊會設的人諮詢。

10 javac xxx.java順利通過了,但是java xxx的時候顯示什麼“NoClassDefFoundError”。
 答:你遇到了classpath問題。java命令在一定的範圍(classpath)內搜索你要用的class文件,但是未能找到。首先請確認你沒有錯敲成java xxx.class,其實你並不需要設置該變量,但如果你設置了該變量又沒有包含.(代表當前目錄)的項,你就會遇到這個問題。請在你的CLASSPATH環境變量中加入一項. 或乾脆刪掉這個變量。如果你使用了並非JDK自帶的標準包,比如javax.servlet.*包,也會遇到這個問題,請將相應的jar文件加入classpath。如果你在java源文件中定義了package,請參見15。


11 我在java xxx的時候顯示"Exception in thread "main" java.lang.NoSuchMethodError: main"。
 答:首先,在你的程序中每個java文件有且只能有一個public類,這個類的類名必須和文件名的大小寫完全一樣。其次,在你要運行的類中有且只能有一個public static void main(String[] args)方法,這個方法就是你的主程序。

12 package是什麼意思?怎麼用?
 答:爲了唯一標識每個類並分組,java使用了package的概念。每個類都有一個全名,例如String的全名是java.lang.String,其中java.lang是包名,String是短名。這樣,如果你也定義了String,你可以把它放在mypackage中,通過使用全名mypackage.String和java.lang.String來區分這兩個類。同時,將邏輯上相關的類放在同一個包中,可以使程序結構更爲清楚。

 你要做的就是在java文件開頭加一行"package mypackage;"。注意包沒有嵌套或包含關係,A包和A.B包對java命令來說是並列的兩個包。
13 我沒有聲明任何package會怎麼樣?
 答:你的類被認爲放在默認包中。這時全名和短名是一致的。

14 在一個類中怎麼使用其他類?
 答:如果你使用java.lang包中的類,不用做任何事。如果你使用其他包中的類,使用import package1.class1; 或 import package2.*;這裏.*表示引入這個包中的所有類。然後在程序中你可以使用其他類的短名。如果短名有衝突,使用全名來區分。

15 我用了package的時候顯示"NoClassDefFoundError",但是我把所有package去掉的時候能正常運行。
 答:將你的java文件按包名存放。
 比如你的工作目錄是/work,你的類是package1.class1,那麼將它存放爲/work/package1/class1.java。如果沒有聲明包,那麼直接放在/work下。在/work下執行javac package1/class1.java,再執行java package1.class1,你會發現一切正常。另外,你可以考慮開始使用IDE。

16 我想把java編譯成exe文件,該怎麼做?
 答:JDK只能將java源文件編譯爲class文件。class文件是一種跨平臺的字節碼,必須依賴平臺相關的JRE來運行。Java以此來實現跨平臺,有些開發工具可以將java文件編譯爲exe文件。作者反對這種做法,因爲這樣就取消了跨平臺性。如果你確信你的軟件只在Windows平臺上運行,你可以考慮使用C++/C#來編程。

17 我在編譯的時候遇到什麼"deprecated API",是什麼意思?
 答:所謂deprecated是指已經?時,但是爲了向前兼容起見仍然保留的方法,這些方法可能會在以後取消支持。你應當改用較新的方法。一般在API裏面會說明你應當用什麼方法來代替之。


三、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。如果你要輸入輸出雙字節文字比如中文,請使用作者的做法。
22 我怎麼從文件輸入一個int/double/字符串?
 答:類似於從鍵盤輸入,只不過換成

 BufferedReader fin = new BufferedReader( new FileReader(" myFileName " ) );
 PrintWriter fout = new PrintWriter( new FileWriter(" myFileName " ) );


 另外如果你還沒下載API,請開始下載並閱讀java.io包中的內容。

23 我想讀寫文件的指定位置,該怎麼辦?
 答:你肯定沒有認真看API。java.io.RandomAccessFile可以滿足你的需要。

24 怎麼判斷要讀的文件已經到了盡頭?
 答:你肯定沒有認真看API。在Reaer的read方法中明確說明返回-1表示流的結尾。


四、 關鍵字篇

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篇

48 java能動態分配數組嗎?
 答:可以。例如int n = 3; Language[] myLanguages = new Language[n];

49 我怎麼知道數組的長度?
 答:用length屬性。如上例中的 myLanguages.length 就爲 3。

50 我還想讓數組的長度能自動改變,能夠增加/刪除元素。
 答:用順序表--java.util.List接口。你可以選擇用ArrayList或是LinkedList,前者是數組實現,後者是鏈表實現。例如:

 List list = new ArrayList();


 或是

 List list = new LinkedList();



51 什麼是鏈表?爲什麼要有兩種實現?
 答:請補習數據結構。

52 我想用隊列/棧。
 答:用java.util.LinkedList。

53 我希望不要有重複的元素。
 答:用集合--java.util.Set接口。例如:Set set = new HashSet()。

54 我想遍歷集合/Map。
 答:用java.util.Iterator。參見API。

55 我還要能夠排序。
 答:用java.util.TreeSet。例如:Set set = new TreeSet()。放進去的元素會自動排序。你需要爲元素實現Comparable接口,還可能需要提供equals()方法,compareTo()方法,hashCode()方法。

56 但是我想給數組排序。
 答:java.util.Arrays類包含了sort等實用方法。

57 我想按不同方法排序。
 答:爲每種方法定義一個實現了接口Comparator的類並和Arrays綜合運用。

58 Map有什麼用?
 答:存儲key-value的關鍵字-值對,你可以通過關鍵字來快速存取相應的值。

59 set方法沒問題,但是get方法返回的是Object。
 答:強制類型轉換成你需要的類型。參見45。

60 我要獲得一個隨機數。
 答:使用java.util.Random類。

61 我比較兩個String總是false,但是它們明明都是"abc" !
 答:比較String一定要使用equals或equalsIgnoreCase方法,不要使用 == !
 ==比較的是兩個引用(變量)是否指向了同一個對象,而不是比較其內容。

62 我想修改一個String但是在String類中沒找到編輯方法。
 答:使用StringBuffer類。

 String str = "......."; //待處理的字符串
 StringBuffer buffer = new StringBuffer(str); //使用該字符串初始化一個
 StringBuf
 fer
 buffer.append("..."); //調用StringBuffer的相關API來編輯字符串
 String str2 = buffer.toString(); //獲得編輯後的字符串


 另外,如果你需要將多個字符串連接起來,請儘量避免使用+號直接連接,而是使用StringBuffer.append()方法。

63 我想處理日期/時間。
 答:使用java.util.Date類。你可以使用java.text.SimpleDateFormat類來在String和Date間互相轉換。

 SimpleDateFormat formatter = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss"); //規定日期格式
 Date date = formatter.parse("2003-07-26 18:30:35"); //將符合格式的String轉換爲Date
 String s = formatter.format(date); //將Date轉換爲符合格式的String


 關於定義日期格式的詳細信息請參見API。

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章