Effective C#:改善C#代碼的50個有效方法(原書第3版)
定 價:79 元
叢書名:Effective系列叢書
- 作者:[美]比爾·瓦格納(Bill Wagner)
- 出版時間:2018/5/1
- ISBN:9787111597193
- 出 版 社:機械工業(yè)出版社
- 中圖法分類:TP312C
- 頁碼:251
- 紙張:膠版紙
- 版次:1
- 開本:16開
本書分為5個章節(jié),第1章介紹一些常見的語言結(jié)構(gòu),它們是開發(fā)者手頭必備的工具,無論創(chuàng)建什么樣的類型與算法,都離不開它們;第2章介紹一些設(shè)計習(xí)慣,告訴你應(yīng)該怎樣把代碼寫得與托管環(huán)境相協(xié)調(diào);第3章講解怎樣用泛型取代System.Object以及強制類型轉(zhuǎn)換,并討論一些高級技術(shù),如約束、泛型特化、方法約束以及向后兼容等;第4章講解LINQ、查詢語法以及與之相關(guān)的語言特性,包括什么時候運用擴展方法把協(xié)定與實現(xiàn)相分離、怎樣有效地使用閉包以及如何編寫匿名類型等;第5章指引你把C#程序中的異常與錯誤處理好。
Preface 前 言本書第1版于2004年出版,到了2016年,C#開發(fā)社群的情況已經(jīng)有了很大的變化。使用這門語言編寫程序的人越來越多,很多人現(xiàn)在都把C#當(dāng)作首選的工作語言,并且不會再按照使用其他語言時所形成的那些習(xí)慣來使用這門語言。此外,C# 開發(fā)者所具備的經(jīng)驗各不相同,從剛畢業(yè)的學(xué)生到擁有數(shù)十年經(jīng)驗的專業(yè)開發(fā)者,都有人在用C#寫程序,而且 C# 所支持的平臺也比原來更加廣泛。你可以用它架設(shè)服務(wù)器或制作網(wǎng)站,也可以為各種操作系統(tǒng)開發(fā)桌面版本或移動版本的應(yīng)用程序。
這次升級的第3版既考慮到C#語言本身的變化,也考慮到使用這門語言的人(或者說 C# 開發(fā)社群)所發(fā)生的變化。筆者并不打算講述C#語言的演變歷程,而是關(guān)注怎樣用好當(dāng)前版本的 C# 語言。舊版的某些條目已經(jīng)與當(dāng)今的 C# 語言或 C# 應(yīng)用程序脫節(jié)了,這些內(nèi)容不會出現(xiàn)在新版中。新版中會添加一些條目,以講述 C# 語言的新特性與 .NET 框架的新功能,這些內(nèi)容是從軟件產(chǎn)品的開發(fā)過程中提煉出來的,許多C#開發(fā)者采用這些特性開發(fā)了多個版本的軟件?催^《More Effective C#》第1版的讀者稍后可能會發(fā)現(xiàn),那本書里的某些內(nèi)容已經(jīng)移到了本書中。在本書第3版中,筆者重新編排《More Effective C#》的內(nèi)容,刪除了原有的許多條目,以便在那本書的第2版中添加其他一些條目?傊@本書里的 50 個條目都是一些編程建議,可以幫助你更為高效地使用 C# 語言,從而成為更加專業(yè)的開發(fā)者。
本書預(yù)設(shè)的語境是 6.0 版本的 C#,然而筆者并不會把該版本的功能全都拿出來講。與Effective Software Development系列的其他書一樣,這本書所關(guān)注的也是怎樣用語言特性來解決日常工作中可能遇到的問題,并提供實用的建議。在 C# 6.0 版的這些特性中,筆者會特意挑出一些來講,因為其中的某些特性能夠使開發(fā)者以更好的方式來編寫常用的代碼。網(wǎng)上搜到的寫法可能是針對許多年前的C#版本而寫的,有了新版C#所引入的特性之后,開發(fā)者就可以用更好的寫法來完成那些任務(wù)了,對于此類情況,筆者會專門指出。
書中的很多建議都可以用Roslyn平臺的Analyzer及Code Fix加以體現(xiàn),從而驗證開發(fā)者所寫的代碼是否符合這些建議。筆者把相關(guān)的Analyzer放在了這里:https://github.com/BillWagner/EffectiveCSharpAnalyzers。你可以提交 issue,以表達自己的看法,或是發(fā)送 pull request為項目添加新的內(nèi)容。
讀者對象本書面向的是那些使用 C# 來完成日常工作的職業(yè)開發(fā)者。由于本書假設(shè)讀者已經(jīng)熟悉了 C# 的語法及語言特性,因此,并不會按部就班地講解這些特性,而是會告訴你應(yīng)該怎樣把當(dāng)前這一版 C# 語言所擁有的各種特性融入日常的開發(fā)工作中。
除了要熟悉語言本身的特性之外,還應(yīng)該對 CLR(Common Language Runtime,公共語言運行時)及 JIT(Just-In-Time,即時)編譯器有所了解。
內(nèi)容提要有一些語言結(jié)構(gòu)是每次寫 C# 程序時幾乎都會用到的,這些常見的寫法出現(xiàn)在本書的第1章中,它們是開發(fā)者手頭必備的工具,無論創(chuàng)建什么樣的類型與算法,都離不開這些工具。
盡管 C# 程序運行在托管環(huán)境中,但并不是說開發(fā)者什么事情都不用操心。要想令程序的性能滿足需求,就必須編寫出能夠與托管環(huán)境相協(xié)調(diào)的代碼,這不是單靠性能測試與性能調(diào)整就可以實現(xiàn)的。因此,第2章會介紹一些設(shè)計習(xí)慣,告訴你應(yīng)該怎樣把代碼寫得與托管環(huán)境相協(xié)調(diào)。以良好的設(shè)計風(fēng)格為基礎(chǔ),可以更加有效地優(yōu)化細節(jié)問題。
自C# 2.0以來所引入的很多新技術(shù)都是以泛型為依托的。第3章講解怎樣用泛型取代System.Object以及強制類型轉(zhuǎn)換,然后,筆者會討論一些高級技術(shù),例如約束、泛型特化、方法約束以及向后兼容等。讀完本章之后,你會學(xué)到很多泛型技巧,從而能夠更加順暢地表達出自己的設(shè)計思路。
第4章會講解 LINQ、查詢語法以及與之相關(guān)的語言特性。你會了解到在哪些情況下應(yīng)該運用擴展方法把協(xié)定與實現(xiàn)相分離,還會學(xué)到應(yīng)該怎樣有效地使用閉包以及如何編寫匿名類型。此外,筆者還會解釋編譯器怎樣把查詢關(guān)鍵字映射成方法調(diào)用、如何區(qū)分委托與表達式樹以及必要時怎樣在二者之間轉(zhuǎn)換,以及如何對查詢做出轉(zhuǎn)義以獲取純量形式的結(jié)果。
第 5 章會指引你把 C# 程序中的異常與錯誤處理好。筆者要講解怎樣確保程序中的錯誤能夠得到適當(dāng)?shù)膮R報,以及如何令程序的狀態(tài)在出錯之后依然保持穩(wěn)定,甚至與出錯之前一樣。此外,你還會學(xué)到怎樣給使用代碼的人提供便利,令他們能夠更加順暢地調(diào)試你所編寫的程序。
代碼約定要想把范例代碼印在書中,就必須在保持清晰的前提下顧及篇幅。筆者盡量把代碼寫得簡短,以凸顯其中最關(guān)鍵的部分,并把類或方法中的其他部分省掉。有時為了節(jié)省篇幅,還會把錯誤恢復(fù)代碼也省掉。public 方法自然應(yīng)該驗證其參數(shù)以及外界輸入給它的數(shù)據(jù),但考慮到篇幅,筆者通常會把這些代碼去掉。此外,很多復(fù)雜的算法還會對方法調(diào)用做出核查,而且會包含try/finally子句,這些代碼也因同樣的理由而刪去。
常見的命名空間就不再寫出了。你可以認為每一份范例代碼前面都寫有下面幾條 using語句:
提供反饋意見筆者與本書的審閱者都盡力確保書中的內(nèi)容正確無誤,盡管如此,本書與范例代碼里面可能還是會有一些錯誤,讀者如果發(fā)現(xiàn)某個地方寫錯了,請發(fā)郵件至bill@thebillwagner.com,或通過Twitter號碼@billwagner聯(lián)系我?闭`表將會發(fā)布至http://thebillwagner.com/Resources/Effectivecs。書中的很多條目是筆者在與其他C#開發(fā)者通過電子郵件及Twitter討論之后寫出的,讀者若對這些編程建議有疑問或意見,也請聯(lián)系筆者。更為一般的話題可參見筆者博客:http://thebillwagner.com/blog。
致謝我要感謝為本書做出貢獻的諸多人士。很榮幸能在這些年里與大家一起使用 C# 語言。C# Insiders 郵件列表中的每位朋友(無論身處 Microsoft 公司之內(nèi)或之外)都為本書提供了創(chuàng)意,并且愿意與我交流,使我能把這本書寫得更好。
必須特別感謝下面這幾位 C# 開發(fā)者:Jon Skeet、Dustin Campbell、Kevin Pilch-Bisson、Jared Parsons、Scott Allen 以及 Mads Torgersen。感謝你們與我溝通、向我提供意見,并將其轉(zhuǎn)變?yōu)榫唧w的成果。這一版的很多新想法都是根據(jù)諸位的意見而形成的。
這一版的技術(shù)評審團隊同樣很出色。Jason Bock、Mark Michaelis 與 Eric Lippert 仔細閱讀了文稿與范例代碼,以確保讀者能拿到一本優(yōu)質(zhì)的書籍。他們的水平相當(dāng)高,不僅全面而徹底地審閱了本書,而且還提供了一些建議,幫助我把其中的很多話題解釋得更為清楚。
我與Addison-Wesley出版社的編輯團隊合作得相當(dāng)愉快。Trina Macdonald 是一位優(yōu)秀的編輯,總能督促我把書寫好。Mark Renfro 與 Olivia Basegio 是她的得力幫手,我依靠他們完成了很多工作,這本書的定稿能夠達到現(xiàn)在這樣的質(zhì)量,與他們的努力有很大關(guān)系,從頭到尾的每一頁內(nèi)容都是如此。Curt Johnson 致力于發(fā)售這本技術(shù)圖書,無論是哪種格式都有他的一份心力在里面。
感謝Scott Meyers再度將本書收入 Effective 書系,他閱讀了整部文稿,并提出了一些改進建議。Meyers 雖然不是做 C# 的,但卻有著豐富的軟件開發(fā)經(jīng)驗,能夠把文稿中沒有解釋清楚的地方找出來,而且能指出其中有哪些技巧還不足以總結(jié)成心得推薦給大家使用。他的意見,給我?guī)砹撕艽蟮膸椭?br> 感謝家人留出時間,令我可以寫完這本書。我花了很長時間撰寫書稿并制作范例代碼,妻子 Marlene 總是給予我支持。有她的鼓勵,我才能把這本書和其他的書寫好。
Contents 目 錄
本書贊譽
譯者序
前言
第1章 C#語言的編程習(xí)慣1
第1條:優(yōu)先使用隱式類型的局部變量1
第2條:考慮用readonly代替const8
第3條:優(yōu)先考慮is或as運算符,盡量少用強制類型轉(zhuǎn)換12
第4條:用內(nèi)插字符串取代string.Format()20
第5條:用FormattableString取代專門為特定區(qū)域而寫的字符串24
第6條:不要用表示符號名稱的硬字符串來調(diào)用 API26
第7條:用委托表示回調(diào)28
第8條:用null條件運算符調(diào)用事件處理程序31
第9條:盡量避免裝箱與取消裝箱這兩種操作34
第10條:只有在應(yīng)對新版基類與現(xiàn)有子類之間的沖突時才應(yīng)該使用 new修飾符38
第2章 .NET的資源管理42
第11條:理解并善用 .NET的資源管理機制42
第12條:聲明字段時,盡量直接為其設(shè)定初始值47
第13條:用適當(dāng)?shù)姆绞匠跏蓟愔械撵o態(tài)成員50
第14條:盡量刪減重復(fù)的初始化邏輯52
第15條:不要創(chuàng)建無謂的對象60
第16條:絕對不要在構(gòu)造函數(shù)里面調(diào)用虛函數(shù)64
第17條:實現(xiàn)標(biāo)準(zhǔn)的dispose 模式67
第3章 合理地運用泛型74
第18條:只定義剛好夠用的約束條件76
第19條:通過運行期類型檢查實現(xiàn)特定的泛型算法82
第20條:通過IComparable及IComparer定義順序關(guān)系88
第21條:創(chuàng)建泛型類時,總是應(yīng)該給實現(xiàn)了IDisposable的類型參數(shù)提供支持95
第22條:考慮支持泛型協(xié)變與逆變98
第23條:用委托要求類型參數(shù)必須
提供某種方法104
第24條:如果有泛型方法,就不要
再創(chuàng)建針對基類或接口的
重載版本110
第25條:如果不需要把類型參數(shù)所
表示的對象設(shè)為實例字段,
那么應(yīng)該優(yōu)先考慮創(chuàng)建泛
型方法,而不是泛型類114
第26條:實現(xiàn)泛型接口的同時,還
應(yīng)該實現(xiàn)非泛型接口118
第27條:只把必備的契約定義在接
口中,把其他功能留給擴
展方法去實現(xiàn)124
第28條:考慮通過擴展方法增強已
構(gòu)造類型的功能128
第4章 合理地運用LINQ131
第29條:優(yōu)先考慮提供迭代器方法,
而不要返回集合131
第30條:優(yōu)先考慮通過查詢語句來
編寫代碼,而不要使用循環(huán)
語句137
第31條:把針對序列的API設(shè)計得
更加易于拼接142
第32條:將迭代邏輯與操作、謂詞
及函數(shù)解耦149
第33條:等真正用到序列中的元素
時再去生成153
第34條:考慮通過函數(shù)參數(shù)來放松
耦合關(guān)系155
第35條:絕對不要重載擴展方法162
第36條:理解查詢表達式與方法調(diào)
用之間的映射關(guān)系165
第37條:盡量采用惰性求值的方式
來查詢,而不要及早求值177
第38條:考慮用lambda表達式來
代替方法182
第39條:不要在Func與Action中
拋出異常186
第40條:掌握盡早執(zhí)行與延遲執(zhí)行
之間的區(qū)別188
第41條:不要把開銷較大的資源捕
獲到閉包中193
第42條:注意IEnumerable與
IQueryable形式的數(shù)據(jù)
源之間的區(qū)別206
第43條:用Single()及First()
來明確地驗證你對查詢結(jié)果
所做的假設(shè)211
第44條:不要修改綁定變量214
第5章 合理地運用異常220
第45條:考慮在方法約定遭到違背
時拋出異常220
第46條:利用using與try/finally
來清理資源224
第47條:專門針對應(yīng)用程序創(chuàng)建異常231
第48條:優(yōu)先考慮做出強異常保證237
第49條:考慮用異常篩選器來改寫
先捕獲異常再重新拋出的
邏輯244
第50條:合理利用異常篩選器的副
作用來實現(xiàn)某些效果248
中英文詞匯對照表252