《java編程思想》讀書(shū)筆記
這是一份試圖提綱挈領(lǐng)的讀書(shū)筆記,《java編程思想》這本八百多頁(yè)的書(shū)娓娓道來(lái)地包含了太多細(xì)節(jié),這對(duì)讀者是非常貼心的,我也強(qiáng)烈建議細(xì)細(xì)讀這本書(shū),如果你希望在短時(shí)間內(nèi)學(xué)會(huì)java這種語(yǔ)言,那么這本書(shū)不是最好的選擇,你可以看看譚浩強(qiáng)系列。我把看這本書(shū)的過(guò)程中,個(gè)人覺(jué)得每一章中最重要的思想、用整理在這里,希望自己回顧的時(shí)候有所參照和提高。也希望大家?guī)е瑯拥哪康膩?lái)看本篇讀書(shū)筆記。
第一章 對(duì)象導(dǎo)論
比起過(guò)程型語(yǔ)言編寫(xiě)的程序,用面向?qū)ο笳Z(yǔ)言編寫(xiě)的程序更加簡(jiǎn)單、易于理解、可復(fù)用!禼++編程思想》里也有這一章,是一個(gè)拋磚引自己的玉的章節(jié),不明白可以跳過(guò),回頭再看。
第二章 一切都是對(duì)象
java語(yǔ)言里面,一切都是對(duì)象,并且程序員是通過(guò)引用來(lái)操縱對(duì)象。一個(gè)簡(jiǎn)單的例子,非常輕松地讓讀者進(jìn)入java的世界。需要注意的是java數(shù)據(jù)會(huì)儲(chǔ)存在5個(gè)不同的地方:寄存器、堆棧、堆、常量存儲(chǔ)、非ram存儲(chǔ),用new創(chuàng)建的一般對(duì)象都放在堆中,而特殊的基本對(duì)象放在堆棧中,如果想把基本對(duì)象也放在堆中,需要包裝基本類型。
第三章 操作符
java中的操作符語(yǔ)法類似于c,所以學(xué)習(xí)起來(lái)一點(diǎn)困難也沒(méi)有。要特別注意兩個(gè)比較大的整數(shù)相加或者相乘的時(shí)候的溢出問(wèn)題,用long或者biginteger解決這個(gè)問(wèn)題。
第四章 控制執(zhí)行流程
我想起《pointer on c》這本書(shū)第一章就有這一句話,本書(shū)適合那些希望迅速學(xué)習(xí)一門(mén)新語(yǔ)言而不是被“為什么if和for很重要”的弱智問(wèn)題耽擱進(jìn)度的讀者。呵呵,這一章很不厭其煩地介紹了運(yùn)算、操作符優(yōu)先級(jí)、類型轉(zhuǎn)換、選擇循環(huán)等基本特性,有c或者c++編程經(jīng)驗(yàn)的讀者可以大概瀏覽一下。
第五章 初始化和清理
關(guān)于初始化:
1.初始化很重要,一定不要忘記。而且java編譯器會(huì)很好的防止使用未初始化數(shù)據(jù)的意外,這是比c和c++更優(yōu)的地方。
2.編譯器初始化的順序?yàn)椋?/p>
a.類首次加載的時(shí)候,有關(guān)靜態(tài)初始化的所有動(dòng)作都會(huì)執(zhí)行。
a1.類的加載包括首次創(chuàng)建該類型的對(duì)象,或者該類的靜態(tài)方法/靜態(tài)域首次被訪問(wèn)
a2.靜態(tài)域的初始化在一切初始化之前,即靜態(tài)變量散布在代碼不同的地方,它們也會(huì)在任何方法(包括構(gòu)造器)調(diào)用之前被初始化
b.當(dāng)用new calssname()創(chuàng)建對(duì)象的時(shí)候,會(huì)在堆上開(kāi)辟足夠的存儲(chǔ)空間,這塊存儲(chǔ)空間被清零,然后執(zhí)行字段的初始化動(dòng)作。(這里的字段初始化都是非靜態(tài)的,因?yàn)殪o態(tài)的變量已經(jīng)在a中執(zhí)行完畢,而且靜態(tài)變量存儲(chǔ)在不同的地方,靜態(tài)數(shù)據(jù)只占用一份存儲(chǔ)空間)
c.執(zhí)行構(gòu)造器
關(guān)于清理
c++關(guān)于清理的部分包含很大不確定性。目前需要知道的事情是,正常情況下,我們是不需要調(diào)用finalize方法的,而且垃圾回收區(qū)會(huì)自動(dòng)回收不再使用的對(duì)象,同時(shí)我們需要自己注意一些需要關(guān)閉的文件。
需要注意的是,用=對(duì)數(shù)組進(jìn)行“賦值”的時(shí)候,實(shí)際上是引用的傳遞,就是說(shuō),二者指向同一堆。
第六章 訪問(wèn)權(quán)限控制
關(guān)于包
你應(yīng)該有一個(gè)自己的域名,這樣發(fā)布你的java程序的時(shí)候,就可以將你的包名設(shè)置為你的域名倒轉(zhuǎn)。想要正確讓包工作,要正確設(shè)置classpath,對(duì)于新手來(lái)說(shuō),這的確是一個(gè)挑戰(zhàn)。我當(dāng)初就難到了。
關(guān)于訪問(wèn)權(quán)限修飾詞
值得注意的是,如果兩個(gè)編譯單元放在同一個(gè)目錄下并且都沒(méi)有設(shè)置包名的話,他們對(duì)于對(duì)方都是擁有包訪問(wèn)權(quán)限的。訪問(wèn)權(quán)限修飾詞是修飾方法和數(shù)據(jù),而不是類。類只有兩種訪問(wèn)權(quán)限,包訪問(wèn)權(quán)限或public訪問(wèn)權(quán)限。默認(rèn)為包訪問(wèn)權(quán)限。如果不希望其它任何人對(duì)該類擁有訪問(wèn)權(quán)限,可以把所有的構(gòu)造器設(shè)置為private。但是有一個(gè)例外,可以通過(guò)該類自己的static成員內(nèi)部創(chuàng)建(于是就有了工廠設(shè)計(jì)模式和單例設(shè)計(jì)模式)。
第七章 復(fù)用類
有三種方法復(fù)用類:組合,繼承,代理。
組合即是在新的類里面放上已經(jīng)定義的類的對(duì)象,然后通過(guò)調(diào)用它的方法來(lái)實(shí)現(xiàn)自己的功能。
繼承是通過(guò)extends關(guān)鍵詞繼承某一父類,這樣就能訪問(wèn)父類的所有public方法(因此為了繼承,一般的規(guī)則是將父類的所有數(shù)據(jù)成員都指定為private,將所有的方法都指定為public)。子類的初始化需要注意的是,(當(dāng)創(chuàng)建了一個(gè)子類的對(duì)象時(shí),該對(duì)象包含一個(gè)基類的子對(duì)象)java會(huì)在子類的構(gòu)造器中插入對(duì)基類默認(rèn)構(gòu)造器的調(diào)用。但是如果沒(méi)有默認(rèn)的基類構(gòu)造器,或者想調(diào)用一個(gè)帶參數(shù)的基類構(gòu)造器,就必須用關(guān)鍵詞super顯式地編寫(xiě)調(diào)用基類構(gòu)造器的語(yǔ)句,并且配上適當(dāng)?shù)膮?shù)列表。
代理很有意思,(我們姑且使用導(dǎo)出類和基類這樣的字眼,但要清楚我們不是在討論繼承里面的關(guān)鍵詞)在導(dǎo)出類里保存一個(gè)基類的對(duì)象,然后用自己的方法對(duì)該基類的種種方法進(jìn)行包裝。
如何決定使用哪種方法復(fù)用類呢?is-a就繼承,has-a就用組合。而且,組合比繼承總體上使用更廣泛、代價(jià)更小。
向上轉(zhuǎn)型
這個(gè)就牛逼了,第八章,第九章,第十章都與此密切相關(guān)?赐瓯緯(shū)之后印象最深的就是向上轉(zhuǎn)型了。
使用final的原因有很多種,一定要弄清楚為什么使用final,是由于設(shè)計(jì)還是效率。
final作用于數(shù)據(jù)的時(shí)候:final作用在基本對(duì)象比如int上,該值就成為不可改變的,一旦被初始化就無(wú)法再被更改,但是作用在普通的對(duì)象引用的時(shí)候,final使引用恒定不變,但是引用指向的對(duì)象是可變的。編譯器需要我們確保final對(duì)象一定要被初始化,我們可以通過(guò)在構(gòu)造器中初始化他們,以達(dá)到相對(duì)自由的效果(稱為空白final,我認(rèn)為這個(gè)名字容易讓人誤解)。java允許在參數(shù)列表中以聲明的方式將參數(shù)指明為final,這一特性主要用來(lái)向匿名內(nèi)部類傳遞數(shù)據(jù)(這很重要)。
final作用于方法的.時(shí)候,說(shuō)明作者想保持該方法在繼承的過(guò)程中不被改變,并且不被覆蓋。同時(shí),被final修飾的方法會(huì)被關(guān)閉“動(dòng)態(tài)綁定”,這樣編譯器就會(huì)為final方法調(diào)用生成“有限”有效的代碼。之所以說(shuō)有限,是因?yàn)殡S著編譯器的牛逼,它生成的代碼越來(lái)越有效。
final作用于類的時(shí)候,即是作者聲明對(duì)該類的設(shè)計(jì)不允許任何繼承。
學(xué)習(xí)得更深入一些,可能對(duì)以下事實(shí)感到有興趣:java中所有的事物都是對(duì)象,每個(gè)類的編譯代碼都存在于電腦中的文件夾里(文件夾的層次根據(jù)反轉(zhuǎn)域名得到),該文件只有在需要使用程序代碼時(shí)才被加載。具體的說(shuō),就是“類在其任何static成員函數(shù)(包括構(gòu)造函數(shù))被訪問(wèn)時(shí)加載”。
第八章 多態(tài)
多態(tài)的重要基本原理就是向上轉(zhuǎn)型:繼承允許將對(duì)象視為它自己本身的類型或其基類型加以處處理。
將一個(gè)方法調(diào)用和一個(gè)方法主題關(guān)聯(lián)起來(lái)稱為綁定,java中所有的方法都是后期綁定(除了static方法和final方法),所以我們可以編寫(xiě)只與基類打交道的程序代碼,并且這些代碼對(duì)所有的導(dǎo)出類都可以正確運(yùn)行。
(為什么static不動(dòng)態(tài)綁定:因?yàn)閟tatic方法的主要用法就是用類名.方法名這樣的方式來(lái)調(diào)用,不存在“發(fā)送消息給某個(gè)對(duì)象,讓對(duì)象判斷自己怎么做”這樣的情況。
為什么final不動(dòng)態(tài)綁定:這是早期final的一種用法,由程序員指定某方法為final,意味著程序員明了動(dòng)態(tài)綁定的機(jī)制,并且聲明該方法不需要?jiǎng)討B(tài)綁定,這樣可以獲得更好的性能。這種用法已經(jīng)很少使用了。)
初始化的時(shí)候,導(dǎo)出類的構(gòu)造函數(shù)會(huì)自動(dòng)調(diào)用基類的默認(rèn)構(gòu)造函數(shù),此過(guò)程一直遞歸到最基本的基類。如果需要調(diào)用有參數(shù)的構(gòu)造函數(shù)就需要手動(dòng)執(zhí)行。反過(guò)來(lái),如果需要進(jìn)行清理工作(大部分時(shí)候我們都不需要),務(wù)必手動(dòng)執(zhí)行基類的清理工作先。比如繼承鏈的每個(gè)類都實(shí)現(xiàn)dispose()方法,那么執(zhí)行某個(gè)類的清理工作的時(shí)候,需要手動(dòng)調(diào)用super.dispose()。不過(guò)此種情況下,務(wù)必在執(zhí)行super.dispose()之前釋放成員對(duì)象,清理順序與執(zhí)行順序是相反的。
此外,構(gòu)造器方面有更加復(fù)雜的調(diào)用機(jī)制,我們不用理它,只需要知道一條有效的準(zhǔn)則“用盡可能簡(jiǎn)單的方法使對(duì)象進(jìn)入正常狀態(tài),如果可以的話避免調(diào)用其它方法”。
java編譯器能夠允許向上多態(tài),就是因?yàn)閖ava的機(jī)制能保存對(duì)象的類型信息,即rtti,正因?yàn)檫@種機(jī)制,java編譯器也允許向下轉(zhuǎn)型,以獲得擴(kuò)展類的“擴(kuò)展出”的方法。(另,擴(kuò)展類“擴(kuò)展”了方法的這種繼承不是“純繼承”,這樣做好不好?用戶自己度量)。向下轉(zhuǎn)型失敗的話會(huì)拋出一個(gè)classcastexception。
雖然這一章都是在講多態(tài),但是多態(tài)并不總是解決問(wèn)題最好的方案,它有可能使事情不必要地復(fù)雜起來(lái),我們應(yīng)該總是優(yōu)先考慮更加靈活的組合。
第九章 接口
一種專門(mén)提供“接口”的類叫抽象類,若含有至少一個(gè)abstract方法,該類就必須被聲明為abstract的。抽象方法沒(méi)有方法體,派生類必須實(shí)現(xiàn)它,否則派生類也必須被生命為抽象的。
interface關(guān)鍵詞使抽象的概念更進(jìn)了一步:1.這個(gè)“類”完全抽象。2.一個(gè)類可以向上轉(zhuǎn)型為多種interface。要讓一個(gè)類遵循某個(gè)特定接口,需要使用implement關(guān)鍵字。
在這一章中出現(xiàn)了“策略設(shè)計(jì)模式”這個(gè)詞。創(chuàng)建一個(gè)能夠根據(jù)所傳遞的參數(shù)對(duì)象的不同而具有不同行為的方法,被稱為策略設(shè)計(jì)模式。
策略設(shè)計(jì)模式跟適配器設(shè)計(jì)模式聯(lián)合使用可以提供非常強(qiáng)大的功能,比如我們遇到了無(wú)法更改的類(別人編寫(xiě)的),想要它滿足我們的接口然后放到設(shè)計(jì)模式里面去(當(dāng)然滿足了接口之后的用法就不止如此了),就可以編寫(xiě)一個(gè)適配器,包裝該類同時(shí)產(chǎn)生我所需要的接口。
使用抽象類和接口的兩個(gè)原因是:1.在多重繼承關(guān)系中(這真的很常見(jiàn),看看java api就知道了),導(dǎo)出類可以被向上轉(zhuǎn)型為每一個(gè)接口。2.防止客戶端程序員創(chuàng)建該類的對(duì)象。那么我們?cè)撌褂贸橄箢愡是接口呢?事實(shí)上,如果知道某事物應(yīng)該成為一個(gè)基類,那么第一選擇應(yīng)該是使它成為一個(gè)接口。
接口之間的繼承能夠形成很好的體系,更像我們的現(xiàn)實(shí)生活。但是要特別注意的是,在不同接口中使用相同的方法名通常會(huì)造成代碼可讀性的混亂,令人不快。
工廠方法設(shè)計(jì)模式是又一個(gè)重要的設(shè)計(jì)模式。我們?cè)诖a中增加額外的間接性,一個(gè)重要的原因是想要?jiǎng)?chuàng)建框架。
【《java編程思想》讀書(shū)筆記】相關(guān)文章:
Java編程開(kāi)發(fā)簡(jiǎn)介08-17
Java基本編程技巧11-16
關(guān)于JAVA的XML編程10-08
java網(wǎng)絡(luò)編程基本知識(shí)08-09
Java編程中如何實(shí)現(xiàn)中文排序09-27
JAVA編程面試題及答案09-16