Java基礎(chǔ):從C++轉(zhuǎn)到Java需注意的地方
——
Java源代碼并不是被編譯成為普通的機(jī)器代碼。而是被翻譯成為虛擬機(jī)可以執(zhí)行的代碼。一個(gè)Java解釋器最終執(zhí)行這些代碼。
Java源代碼并不是被編譯成為普通的機(jī)器代碼。而是被翻譯成為虛擬機(jī)可以執(zhí)行的代碼。一個(gè)Java解釋器最終執(zhí)行這些代碼。這其中沒有連接的過程;解釋在需要的時(shí)候動(dòng)態(tài)的加載一些類;
2.Java是完全面向?qū)ο蟮?nbsp;
Java是一種完全面向?qū)ο蟮恼Z言。這意味著你對(duì)任何一個(gè)Java對(duì)象所做的動(dòng)作都是通過一個(gè)方法實(shí)現(xiàn)的。第一點(diǎn)就是,再也沒有沒有主函數(shù)這樣的孤立的東西了。取而代之的是,你必須開始用一個(gè)對(duì)象的看法看待一個(gè)程序,一個(gè)類的對(duì)象。但是這個(gè)對(duì)象又什么對(duì)象呢?大多數(shù)Java程序只是簡單的通過繼承Java基礎(chǔ)類Object來實(shí)現(xiàn)所需要的東西,但是你可以通過創(chuàng)建程序基礎(chǔ)類用于多個(gè)特性相似的應(yīng)用程序來節(jié)省時(shí)間。
嚴(yán)格的面向?qū)ο蟮囊?guī)定意味著理用原有的C/C++代碼不可以直接不加改動(dòng)的使用;系統(tǒng)調(diào)用也是這樣的。C++中,你可以通過在C++正常的命名空間外聲明extern"C"來使用原有的C的過程調(diào)用,包括系統(tǒng)調(diào)用。
在Java中,只有一個(gè)類似的安全回溯的方法,但是并不是十分簡單的方法。你必須定義一個(gè)本地方法,其目的是為C語言提供接口,然后提供連接的介質(zhì)。Java環(huán)境提供了完成這種任務(wù)的工具,但是整個(gè)過程和C++中提供的extern比微不足道,完成使用C++類的過程則更加復(fù)雜,因?yàn)檫@樣會(huì)引入對(duì)C的借口和C函數(shù)和C++成員函數(shù)的問題。
幸運(yùn)的是,許多常用的系統(tǒng)實(shí)用工具函數(shù)已經(jīng)在系統(tǒng)類中的方法中提供出來,但是這些明顯沒有包含經(jīng)過許多年來你所創(chuàng)建的那些類和過程。所以,在你需要的時(shí)候你應(yīng)該去鉆研一下。
3.Java中沒有獨(dú)立的頭文件
在Java中,關(guān)于類的一切東西都被放到一個(gè)單獨(dú)的文件中。方法的位置只可能在一個(gè)地方出現(xiàn),一個(gè)方法的實(shí)現(xiàn)必須在它的定義過程中同時(shí)進(jìn)行。這樣做得優(yōu)點(diǎn)是在實(shí)現(xiàn)程序的時(shí)候不容易因?yàn)槲募姆峭藉e(cuò)誤而失敗,或者獲取到一個(gè)沒有實(shí)現(xiàn)的聲明。類的聲明可以被Java解釋器利用甚至是從一個(gè)編譯過的單元中獲取,所以不再需要有頭文件,只要有編譯過的文件。
這樣做的缺點(diǎn)與我們編程的過程有關(guān)。許多C++程序員喜歡用頭文件來代替文檔。要看一個(gè)成員函數(shù)的接口參數(shù),只需要看頭文件中的聲明即可。你可以經(jīng)常的看頭文件即可了解怎樣去使用這個(gè)類。在Java中,沒有這樣的總結(jié)。因?yàn)閷?shí)現(xiàn)類方法的代碼必須在方法定義的時(shí)候出現(xiàn),而且,對(duì)于一個(gè)單獨(dú)的函數(shù)的代碼來說就經(jīng)常占據(jù)了一整頁乃至更多。這樣,很難通過看Java的代碼就初步了解類是怎樣使用的。你必須為你需要的類準(zhǔn)備足夠多的文檔。不言而喻,再處理非商業(yè)類庫的時(shí)候文檔是極度缺乏的。
在當(dāng)先的Java環(huán)境中提供了兩個(gè)工具來補(bǔ)償這些,javap來打印類標(biāo)識(shí),javadoc為嵌入式程序提供HTML文檔。
4.用Package來分解Java命名空間
在大的C++工程中經(jīng)常遇到的一個(gè)問題是命名空間--怎樣保證工程的一些程序員不會(huì)創(chuàng)建和另一些程序員一樣名字的類?更糟糕的是,供應(yīng)商可能會(huì)提供一個(gè)包含和你的類一樣名字的類的庫。有許多方法可以解決這一問題,但是很可能在問題發(fā)現(xiàn)之前工程已經(jīng)啟動(dòng),改正錯(cuò)誤是需要付出許多痛苦的。
Java通過"Package"這個(gè)概念解決了這個(gè)問題,Package有效地通過通過集合類劃分了命名空間。在不同包內(nèi)的兩個(gè)同名的類仍然是不同的。關(guān)鍵問題就變成了類是否放置到相應(yīng)的包中。
記住,Java并沒有解決命名沖突的問題。擴(kuò)展一個(gè)基類而引起了派生類的沖突。比如說,如果你最喜歡的供應(yīng)商提供了一些類,然后你把它們用做基類并且派生有一個(gè)foo方法的類,當(dāng)供應(yīng)商提供一個(gè)新版本的類的時(shí)候就可能出現(xiàn),如果供應(yīng)商業(yè)也在新類中提供了一個(gè)foo的方法。
5.異常是Java的重要特性
在C++中,異常和異常處理是十分深?yuàn)W的事情;許多C++程序員從沒有處理過它們甚至不知道它們是何物。異常是在正常的過程中出現(xiàn)的未預(yù)料的錯(cuò)誤,因此,它們不會(huì)從方法中返回,或者作為參數(shù)傳入;但是,它們不能被忽略!這里的一個(gè)例子是計(jì)算一個(gè)書的方根的方法。正常的接口形式是將一個(gè)正數(shù)作為參數(shù)傳入方法,然后方法會(huì)返回一個(gè)正實(shí)數(shù)作為結(jié)果,方法可以檢驗(yàn)這些并且在異常產(chǎn)生的時(shí)候拋出異常。在大多數(shù)系統(tǒng)中,程序員并不是必須這樣做,這樣,一個(gè)沒有考慮到的異常可以使程序不正常的退出。
在Java中,異常已經(jīng)成為語言中非常成熟的部分。方法的說明中就包含了異常的信息,程序處理器也強(qiáng)制檢驗(yàn)如果你使用了一個(gè)能夠產(chǎn)生異常的方法,你就必須檢查異常是否發(fā)生。幾乎所有的Java程序員都會(huì)遇到異常的情況,因?yàn)樵S多非常有用的庫中的類都會(huì)拋出異常。處理異常并不難,但是在一些時(shí)候是需要注意的。一個(gè)方法的文檔會(huì)指明方法拋出的異常的類型。如果你忘了,不要緊,編譯器會(huì)提醒你的。
6.字符串不再是字符數(shù)組
Java中包括了一個(gè)字符串的對(duì)象,并且是個(gè)常量。字符串不像字符數(shù)組一樣,雖然可以簡單的從一個(gè)字符數(shù)組構(gòu)造一個(gè)字符串。你應(yīng)該盡可能的用字符串代替字符數(shù)組,因?yàn)樗麄儾粫?huì)因?yàn)檎`操作而被覆蓋。
7.Java限制了常量對(duì)象和方法
在C++中,你可以正式的聲明一個(gè)函數(shù)參數(shù)或者函數(shù)返回值為const類型,這樣可以有效的防止對(duì)參數(shù)或者返回值的不正當(dāng)修改。另外,你可以聲明一個(gè)成員函數(shù)為const,表明它不可以修改任何他操作的對(duì)象。
Java支持常量操作符,只讀變量,這些通過final關(guān)鍵字實(shí)現(xiàn)。但是Java沒有支持強(qiáng)制的使一個(gè)可寫變量在函數(shù)傳遞、返回的過程中變?yōu)橹蛔x。或者定義一個(gè)不操作修改對(duì)象的常量方法。
在Java中,這個(gè)省略帶來的影響和在C++中相比就非常小了,這很大程度上因?yàn)樽址兞亢妥址麛?shù)組的不同,但是這也帶來一個(gè)引起錯(cuò)誤的隱患。特別地,沒有辦法檢驗(yàn)一個(gè)方法是否可以改動(dòng)對(duì)象。
8.Java沒有指針
理解指針的概念是一個(gè)C或C++程序員最難應(yīng)付的問題。指針也是錯(cuò)誤產(chǎn)生的一大根源。Java中沒有指針,對(duì)象的句柄直接作為參數(shù)傳遞,而不是傳遞指針。另外,你必須通過索引使用數(shù)組。這些都不是什么大問題。然而,沒有指針是在寫含有函數(shù)指針或者成員函數(shù)指針的系統(tǒng)的時(shí)候引起很大麻煩。這個(gè)問題在處理回調(diào)函數(shù)的時(shí)候更加顯著。
9.Java沒有參數(shù)化類型
參數(shù)化類型提供了用一段程序處理許多相似程序的方法。一個(gè)例子就是開平方根的方法,它可以對(duì)int或者float操作。在C++中,這一特性是由模板提供的。
Java中不包含C++中的模板的等價(jià)物。如果你經(jīng)常使用模板來簡化程序,比如說構(gòu)造許多使用相似參數(shù)類型的函數(shù),這簡直就是災(zāi)難。這意味著更多使用復(fù)制、粘貼的過程來手動(dòng)的完成。然而,如果你使用模板來生成類的話,沒有簡單的方法。
10.Java使用垃圾回收
在垃圾回收的語言中,運(yùn)行時(shí)環(huán)境一直監(jiān)測哪些內(nèi)存不被使用。當(dāng)一塊內(nèi)存不用的時(shí)候,系統(tǒng)自動(dòng)的回收內(nèi)存。比如說,一個(gè)對(duì)象在一個(gè)方法中生成,但是沒有被調(diào)用著返回或者沒有儲(chǔ)存為全局變量,不能在方法外部使用。系統(tǒng)自己會(huì)知道哪些變量是你用不到的,哪些是可以用到的。因此,你不必再為破壞對(duì)象回收內(nèi)存而擔(dān)心。在C++中,很多的調(diào)試時(shí)間都被使用到檢查內(nèi)存漏洞中。Java的這種方法很大程度上降低了這種錯(cuò)誤的可能。但是他依然不能處理邏輯混亂的程序,他們不能夠被回收。許多C++的類中的析構(gòu)函數(shù)是用來釋放對(duì)象引用的內(nèi)存的。Java使垃圾回收的事實(shí)說明在Java中不是必需寫析構(gòu)函數(shù)了。但是并不意味著你可以忘記為你的類寫析構(gòu)函數(shù)。比如,一個(gè)對(duì)象打開了網(wǎng)絡(luò)連接就必須被恰當(dāng)?shù)那謇韥黻P(guān)閉這個(gè)連接。在Java中,析構(gòu)函數(shù)被稱作"finalization"方法。
11.Java不支持多重繼承
在任何一個(gè)復(fù)雜的面向?qū)ο蟮南到y(tǒng)中,實(shí)現(xiàn)一個(gè)有更多方法的新類是十分經(jīng)常遇到的事情。比如說,一個(gè)Manager類,需要被作為一個(gè)連表的表頭,但是一個(gè)Manager又必須是一個(gè)Employee。有許多方法來處理這樣的問題。一個(gè)方法是允許從多個(gè)類繼承。在這個(gè)例子中,Manager需要從Linked List和Employee繼承。
Java沒有多重繼承。但是你可以聲明接口--來描述實(shí)現(xiàn)一些功能的編程接口。一個(gè)類可以由多個(gè)接口實(shí)現(xiàn),包括他唯一的功能。不同的類可以由同樣的接口實(shí)現(xiàn)。方法的參數(shù)既可以聲明為類,也可以聲明為接口。如果是接口的話,實(shí)現(xiàn)接口的類就可以作為參數(shù)傳入方法。
接口的概念要比多繼承容易理解一些,但是他有一定的局限性。特別地,你必須在類中實(shí)現(xiàn)接口的時(shí)候編碼去重新實(shí)現(xiàn)類的功能。
12.Java支持多線程
多線程可以使你寫出在同一時(shí)刻完成多種任務(wù)的程序。比如說,你可以在完成讀取一個(gè)大文件之間允許用戶對(duì)已經(jīng)讀取的部分進(jìn)行編輯。你需要把程序分為多線程來執(zhí)行。為安全起見。你的程序要被精心的設(shè)計(jì),因?yàn)榭赡懿恢挂粋€(gè)線程需要對(duì)數(shù)據(jù)進(jìn)行訪問、修改。
Java開始就支持多線程。類和接口用來分解一個(gè)程序成為不同的線程。語言簡單的對(duì)重要的數(shù)據(jù)作同步或者鎖定處理。
13.Java以一些預(yù)定義的類為基礎(chǔ)
默認(rèn)的Java環(huán)境中包括一些從Java基礎(chǔ)類實(shí)現(xiàn)而來的一些包。這些允許你很快的寫出一些有用的程序,這些包如下:
java.awt:當(dāng)今許多應(yīng)用程序都非常依賴GUI,java提供了一個(gè)Abstract Window Toolkid,這可以讓你在不考慮運(yùn)行平臺(tái)的前提下處理GUI對(duì)象。
java.applet:applet的主要目的是提供瀏覽有關(guān)的內(nèi)容。它本身是awt組件的字類并且支持其他一些特性,比如聲音、渲染等。
java.io:java.io提供了對(duì)流、文件、管道的讀寫操作。
java.lang:提供了java的基礎(chǔ)類Objcet,Integar,Float……;
java.net:提供對(duì)網(wǎng)絡(luò)編程的支持。包括處理socket,URL,Internet尋址等。
java.util:為數(shù)據(jù)結(jié)構(gòu)提供的通用實(shí)用工具集。
linux操作系統(tǒng)文章專題:linux操作系統(tǒng)詳解(linux不再難懂)
c++相關(guān)文章:c++教程
評(píng)論