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