前言
你拿起這本書是為了提升你的編程技能。很好,因?yàn)槟阋欢〞?huì)從這本書提供的實(shí)踐知識(shí)中受益。如果你有豐富的C 語(yǔ)言編程經(jīng)驗(yàn),你將學(xué)習(xí)到良好設(shè)計(jì)決策的細(xì)節(jié),以及它們的優(yōu)缺點(diǎn)。如果你對(duì)C 語(yǔ)言編程還比較陌生,你會(huì)找到關(guān)于設(shè)計(jì)決策的指導(dǎo),并且你會(huì)看到這些決策如何一點(diǎn)一點(diǎn)地應(yīng)用到運(yùn)行的代碼示例中,以構(gòu)建更大規(guī)模的程序。
這本書回答了如何構(gòu)建C 程序、如何應(yīng)對(duì)錯(cuò)誤處理、如何設(shè)計(jì)靈活接口等問題。
當(dāng)你對(duì)C 語(yǔ)言編程了解得更多時(shí),經(jīng)常會(huì)出現(xiàn)以下問題:
? 我應(yīng)該返回任何錯(cuò)誤信息嗎?
? 我應(yīng)該使用全局變量errno 來(lái)做這件事嗎?
? 我應(yīng)該有幾個(gè)參數(shù)很多的函數(shù)還是相反?
? 我如何構(gòu)建一個(gè)靈活的接口?
? 我如何構(gòu)建基本的東西,比如迭代器?
對(duì)于面向?qū)ο笳Z(yǔ)言,這些問題大多數(shù)在《設(shè)計(jì)模式:可復(fù)用面向?qū)ο筌浖幕A(chǔ)》一書中得到了很大程度的解答,該書由Erich Gamma、Richard Helm、Ralph Johnson 和John Vlissides 編著(Prentice Hall,1997)。設(shè)計(jì)模式為程序員提供了有關(guān)對(duì)象如何交互以及哪個(gè)對(duì)象擁有哪種其他對(duì)象的最佳實(shí)踐。
此外,設(shè)計(jì)模式展示了這些對(duì)象如何組合在一起。
然而,對(duì)于像C 這樣的過程編程語(yǔ)言,大多數(shù)這些設(shè)計(jì)模式無(wú)法按照四人組描述的方式實(shí)現(xiàn)。C 中沒有原生的面向?qū)ο髾C(jī)制。雖然可以在C 編程語(yǔ)言中模擬繼承或多態(tài),但這可能不是首選,對(duì)于那些使用C 編程且不熟悉面向?qū)ο笳Z(yǔ)言(如C )以及繼承和多態(tài)等概念的程序員,可能更愿意堅(jiān)持他們熟悉的原生C 編程風(fēng)格。然而,采用原生C 編程風(fēng)格,并不是所有面向?qū)ο笤O(shè)計(jì)模式的指導(dǎo)都適用,或者至少對(duì)于非面向?qū)ο缶幊陶Z(yǔ)言,無(wú)法提供設(shè)計(jì)模式中所呈現(xiàn)的特定實(shí)現(xiàn)思路。
我們所面臨的情況是:我們想在C 中進(jìn)行編程,但不能直接應(yīng)用設(shè)計(jì)模式中的大部分知識(shí)。本書展示了如何填補(bǔ)這個(gè)差距,將實(shí)際的設(shè)計(jì)知識(shí)應(yīng)用于C編程語(yǔ)言中。
我為什么寫這本書?
讓我來(lái)告訴你為什么本書中所匯集的知識(shí)對(duì)我來(lái)說非常重要,以及為什么這樣的知識(shí)很難找到。
在學(xué)校里,我學(xué)習(xí)了C 編程作為我的第一門編程語(yǔ)言。就像每個(gè)新的C 程序員一樣,我想知道為什么數(shù)組索引從0 開始,我一開始甚至隨機(jī)嘗試如何放置運(yùn)算符* 和&,來(lái)弄懂C 指針里的魔法到底如何起作用的。
在大學(xué)里,我學(xué)習(xí)了C 語(yǔ)法的實(shí)際工作原理以及它如何在硬件上轉(zhuǎn)化為位和字節(jié)。有了這些知識(shí),我能夠編寫出表現(xiàn)非常好的小程序。然而,我仍然很難理解為什么較長(zhǎng)的代碼看起來(lái)是這個(gè)樣子,我肯定永遠(yuǎn)不會(huì)再提出像以下這樣的解決方案:
typedef struct INTERNAL_DRIVER_STRUCT* DRIVER_HANDLE;
typedef void (*DriverSend_FP)(char byte);
typedef char (*DriverReceive_FP)();
typedef void (*DriverIOCTL_FP)(int ioctl, void* context);
struct DriverFunctions
{
DriverSend_FP fpSend;
DriverReceive_FP fpReceive;
DriverIOCTL_FP fpIOCTL;
};
DRIVER_HANDLE driverCreate(void* initArg, struct DriverFunctions f);
void driverDestroy(DRIVER_HANDLE h);
void sendByte(DRIVER_HANDLE h, char byte);
char receiveByte(DRIVER_HANDLE h);
void driverIOCTL(DRIVER_HANDLE h, int ioctl, void* context);
這樣的代碼引會(huì)發(fā)出許多問題:
? 為什么在結(jié)構(gòu)體中需要函數(shù)指針?
? 為什么函數(shù)需要那個(gè)DRIVER_HANDLE ?
? IOCTL 是什么,為什么我不能使用單獨(dú)的函數(shù)?
? 為什么必須顯式的創(chuàng)建和銷毀函數(shù)?
當(dāng)我開始編寫產(chǎn)品級(jí)的應(yīng)用程序時(shí),這些問題就浮現(xiàn)出來(lái)了。我經(jīng)常遇到一些情況,讓我意識(shí)到我沒有足夠的C 編程知識(shí),例如,我無(wú)法決定如何實(shí)現(xiàn)迭代器,或者決定如何在我的函數(shù)中進(jìn)行錯(cuò)誤處理。我意識(shí)到,盡管我了解C 語(yǔ)法,但我不知道如何應(yīng)用它。我試圖做一些事情,但只是以笨拙的方式或根本無(wú)法實(shí)現(xiàn)。我需要的是關(guān)于如何使用C 編程語(yǔ)言實(shí)現(xiàn)特定任務(wù)的最佳實(shí)踐。例如,我需要知道類似以下的事情:
? 我如何以簡(jiǎn)單的方式獲取和釋放資源?
? 在錯(cuò)誤處理中使用goto 是個(gè)好主意嗎?
? 我是應(yīng)該將接口設(shè)計(jì)得靈活,還是在需要時(shí)直接更改它?
? 我是應(yīng)該使用assert 語(yǔ)句,還是應(yīng)該返回錯(cuò)誤代碼?
? 在C 中如何實(shí)現(xiàn)迭代器?
對(duì)我來(lái)說,非常有趣的一點(diǎn)是,雖然我的有經(jīng)驗(yàn)的同事們對(duì)這些問題有很多不同的答案,但沒有人告訴我哪里有了解這些設(shè)計(jì)決策及其利弊的文檔。
所以接下來(lái)我轉(zhuǎn)向了互聯(lián)網(wǎng),然而我再次感到驚訝:盡管C 編程語(yǔ)言已經(jīng)存在了幾十年,但很難找到這些問題的答案。我發(fā)現(xiàn),雖然有很多關(guān)于C 編程語(yǔ)言基礎(chǔ)和語(yǔ)法的文獻(xiàn),但很少涉及高級(jí)C 編程主題,或者告訴人們?nèi)绾尉帉懩軌驊?yīng)對(duì)產(chǎn)品級(jí)應(yīng)用的優(yōu)美C 代碼。
而這正是這本書的用武之地。本書教會(huì)你如何提升從編寫基本的C 程序,轉(zhuǎn)而到編寫考慮錯(cuò)誤處理的大規(guī)模C 程序,并對(duì)未來(lái)的需求和設(shè)計(jì)變化保持靈活的編程技能。本書使用設(shè)計(jì)模式的概念,逐步為您提供設(shè)計(jì)決策及其利弊。這些設(shè)計(jì)模式被應(yīng)用于運(yùn)行的代碼示例,教會(huì)你為什么先前的示例代碼會(huì)演變成現(xiàn)在的樣子。
這些呈現(xiàn)的模式可以應(yīng)用于任何C 編程領(lǐng)域。由于我來(lái)自多線程實(shí)時(shí)環(huán)境的嵌入式編程領(lǐng)域,因此一些模式可能會(huì)偏向于該領(lǐng)域。不管怎樣,你將會(huì)看到這些模式的通用思想可以應(yīng)用于其他C編程領(lǐng)域,甚至超越了C編程的范圍。
模式基礎(chǔ)
本書中的設(shè)計(jì)指導(dǎo)以模式的形式呈現(xiàn)。將知識(shí)和最佳實(shí)踐以模式的形式呈現(xiàn)的想法源自建筑師克里斯托弗·亞歷山大的《建筑的永恒之道》(牛津大學(xué)出版社,1979)。他使用經(jīng)過驗(yàn)證的小型解決方案來(lái)解決他所在領(lǐng)域的一個(gè)巨大問題:如何設(shè)計(jì)和建造城市。這種應(yīng)用模式的方法被軟件開發(fā)領(lǐng)域采納,其中舉辦了一些關(guān)于模式的會(huì)議,如模式語(yǔ)言程序會(huì)議(PLoP)等,來(lái)用于擴(kuò)展關(guān)于模式的知識(shí)體系。特別是由四人組(Gang of Four)《設(shè)計(jì)模式:可復(fù)用面向?qū)ο筌浖幕A(chǔ)》一書(Prentice Hall,1997),對(duì)軟件開發(fā)人員產(chǎn)生了重要影響,使設(shè)計(jì)模式的概念為軟件開發(fā)人員廣為人知。
但是,什么是模式?市面上有很多定義,如果您對(duì)此主題非常感興趣,那么弗蘭克·布施曼(Frank Buschmann)等編寫的《面向模式的軟件架構(gòu):模式和模式語(yǔ)言》(Wiley,2007)一書可以為你提供準(zhǔn)確的描述和細(xì)節(jié)。在本書中,模式為實(shí)際問題提供了經(jīng)過驗(yàn)證的解決方案。本書中呈現(xiàn)的模式具有的結(jié)構(gòu)如表1 所示。
表1:本書中呈現(xiàn)的模式
模式部分 說明
名字 這是模式的名稱,應(yīng)該容易記住。目標(biāo)是讓程序員在日常語(yǔ)言中使用這個(gè)名稱(就像四人組模式一樣,程序員會(huì)說:抽象工廠創(chuàng)建對(duì)象)。本書中的模式名稱以大寫字母書寫
上下文 上下文部分為模式設(shè)置了背景。它告訴你在哪些情況下可以應(yīng)用這種模式問題 問題部分向您提供了你想要解決的問題的信息。它以粗體字體開始,列出主要的問題陳述,然后詳細(xì)說明為什么這個(gè)問題很難解決(在其他模式格式中,這些詳細(xì)信息放在一個(gè)稱為forces的單獨(dú)部分中)
解決方案 這一部分提供了如何解決問題的指導(dǎo)。它以粗體字體陳述解決方案的主要思想,然后繼續(xù)詳細(xì)說明解決方案。它還通過提供代碼示例來(lái)提供非常具體的指導(dǎo)結(jié)果 這一部分列出了應(yīng)用所描述解決方案的利弊。在應(yīng)用模式時(shí),你應(yīng)該始終確認(rèn)產(chǎn)生的結(jié)果是否符合預(yù)期
已知用例 已知用例提供了所提出的解決方案在實(shí)際應(yīng)用中是有效的證據(jù)。它們還向你展示具體的示例,幫助你理解如何應(yīng)用該模式
以模式的形式呈現(xiàn)設(shè)計(jì)指導(dǎo)的一個(gè)重要優(yōu)勢(shì)是這些模式可以應(yīng)用在各個(gè)場(chǎng)景。如果你面臨的是一個(gè)巨大的設(shè)計(jì)問題,將很難找到一個(gè)確切的指導(dǎo)文檔或解決方案,可以恰好解決這個(gè)問題。相反,你可以將巨大且特定的問題視為許多較小和更通用的問題的總和,并通過逐個(gè)應(yīng)用模式來(lái)逐步解決這些問題。你只需檢查模式所對(duì)應(yīng)的問題描述,選擇并應(yīng)用適合你的問題的模式,并得到一些結(jié)果。這些模式應(yīng)用的結(jié)果可能會(huì)導(dǎo)致另一個(gè)問題,你可以通過接著應(yīng)用另一個(gè)模式來(lái)解決。通過這種方式,逐步地設(shè)計(jì)你的代碼,而不是試圖在編寫第一行代碼之前提前產(chǎn)出一個(gè)完整的設(shè)計(jì)。
如何閱讀這本書?
你應(yīng)該對(duì)C 語(yǔ)言的基礎(chǔ)知識(shí)有了一些了解。你應(yīng)該知道C 語(yǔ)言的語(yǔ)法以及它是如何工作的。例如,這本書不會(huì)教你什么是指針或如何使用它。這本書旨在對(duì)于一些高級(jí)抽象地話題提供提示和指南。
本書中的章節(jié)是獨(dú)立的。你可以按任意順序閱讀它們,并且你可以簡(jiǎn)單地挑選出你感興趣的話題。在接下來(lái)的一段,你會(huì)找到所有模式的概覽,從那里你可以跳轉(zhuǎn)到你感興趣的模式。所以,如果你確切地知道你在尋找什么,直接開始就好了。
如果你不是在尋找某一個(gè)特定的模式,而是想要對(duì)可能的C 程序的設(shè)計(jì)選項(xiàng)有一個(gè)概覽,請(qǐng)通讀本書。在本書中,每章都關(guān)注一個(gè)特定的話題,從如錯(cuò)誤處理和內(nèi)存管理等基本話題開始,然后會(huì)轉(zhuǎn)移到更高級(jí)和特定的話題,如接口設(shè)計(jì)或平臺(tái)獨(dú)立代碼。在每章中,都會(huì)介紹與該話題相關(guān)的模式,以及逐步展示如何應(yīng)用這些模式運(yùn)行代碼示例。
本書的部分展示了兩個(gè)較大的運(yùn)行示例,在這兩個(gè)例子里應(yīng)用了許多的模式。
在這里,你可以學(xué)習(xí)如何通過應(yīng)用模式逐步構(gòu)建一些更加大型的軟件。
OReilly 在線學(xué)習(xí)平臺(tái)(OReilly Online Learning)
近40 年來(lái),OReilly Media 致力于提供技術(shù)和商業(yè)培訓(xùn)、知識(shí)和卓越見解,來(lái)幫助眾多公司取得成功。
公司獨(dú)有的專家和改革創(chuàng)新者網(wǎng)絡(luò)通過OReilly 書籍、文章以及在線學(xué)習(xí)平臺(tái),分享他們的專業(yè)知識(shí)和實(shí)踐經(jīng)驗(yàn)。OReilly 在線學(xué)習(xí)平臺(tái)按照您的需要提供實(shí)時(shí)培訓(xùn)課程、深入學(xué)習(xí)渠道、交互式編程環(huán)境以及來(lái)自O(shè)Reilly 和其他200 多家出版商的大量書籍與視頻資料。更多信息,請(qǐng)?jiān)L問網(wǎng)站:https://www.oreilly.com/。
聯(lián)系我們
任何有關(guān)本書的意見或疑問,請(qǐng)按照以下地址聯(lián)系出版社。
美國(guó):
OReilly Media, Inc.
1005 Gravenstein Highway North
Sebastopol, CA 95472
中國(guó):
北京市西城區(qū)西直門南大街2 號(hào)成銘大廈C 座807 室(100035)
奧萊利技術(shù)咨詢(北京)有限公司
我們?yōu)楸緯渲昧藢倬W(wǎng)頁(yè),在上面列出了勘誤表、示例以及其他附加信息。
你可以通過https://oreil.ly/fluent-c 來(lái)訪問。
如果你對(duì)本書有一些評(píng)論或技術(shù)上的建議,請(qǐng)發(fā)送電子郵件到errata@oreilly.com.cn。
要了解OReilly 的圖書和培訓(xùn)課程的新聞和信息,請(qǐng)?jiān)L問我們的網(wǎng)站https://oreilly.com。
我們的LinkedIn:https://linkedin.com/company/oreilly-media。
我們的Twitter:https://twitter.com/oreillymedia。
我們的YouTube:https://www.youtube.com/oreillymedia。
致謝
我想感謝我的妻子Silke,到現(xiàn)在為止她甚至知道什么是模式,我也想感謝我的女兒Ylvi。她們兩個(gè)都讓我的生活更加快樂,確保我不總是坐在電腦前工作,而是享受生活。
沒有許多模式愛好者的幫助,這本書不可能面世。我想感謝所有在歐洲程序模式語(yǔ)言會(huì)議上的作家工作坊的參與者為我提供關(guān)于模式的反饋。特別是,我想感謝以下人員,在會(huì)議的所謂的牧羊人過程中為我提供了非常有用的反饋:Jari Rauham?ki、Tobias Rauter、Andrea H?ller、James Coplien、Uwe Zdun、Thomas Raser、Eden Burton、Claudius Link、Valentino Vrani和Sumit Kalra。特別感謝我的工作同事,尤其是Thomas Havlovec,他確保我在模式中的C 編程細(xì)節(jié)是正確的。Robert Hanmer、Michael Weiss、David Griffiths 和Thomas Krug 花了很多時(shí)間審查這本書,并為我提供了如何改進(jìn)它的額外想法,非常感謝!也感謝OReilly 的整個(gè)團(tuán)隊(duì),在使這本書成為現(xiàn)實(shí)中給了我很多幫助。特別是,我想感謝我的開發(fā)編輯Corbin Collins 和我的產(chǎn)品編輯Jonathon Owen。
本書的內(nèi)容基于以下在歐洲程序模式語(yǔ)言會(huì)議上被接受并在ACM 發(fā)表的論文。這些論文可以在http://www.preschern.com 網(wǎng)站上免費(fèi)訪問。
? A Pattern Story About C Programming, EuroPLoP 21: 26th European Conference on Pattern Languages of Programs, July 2015, article no. 53,110, https://dl.acm.org/doi/10.1145/3489449.3489978.
? Patterns for Organizing Files in Modular C Programs, EuroPLoP 20:Proceedings of the European Conference on Pattern Languages of Programs,July 2020, article no. 1, 115, https://dl.acm.org/doi/10.1145/ 3424771.3424772.
? Patterns to Escape the #ifdef Hell, EuroPLop 19: Proceedings of the 24th European Conference on Pattern Languages of Programs, July 2019, article no. 2, 112, https://dl.acm.org/doi/10.1145/3361149.3361151.
? Patterns for Returning Error Information in C, EuroPLop 19: Proceedings of the 24th European Conference on Pattern Languages of Programs, July 2019, article no. 3, 114, https://dl.acm.org/doi/10.1145/3361149.3361152.
? Patterns for Returning Data from C Functions, EuroPLop 19: Proceedings of the 24th European Conference on Pattern Languages of Programs, July 2019, article no. 37, 113, https://dl.acm.org/doi/10.1145/3361149.3361188.
? C Patterns on Data Lifetime and Ownership, EuroPLop 19: Proc eedings of the 24th European Conference on Pattern Languages of Programs, July 2019,article no. 36, 113, https://dl.acm.org/doi/10.1145/3361149.3361187.
? Patterns for C Iterator Interfaces, EuroPLoP 17: Proceedings of the 22nd European Conference on Pattern Languages of Programs, July 2017, article no. 8, 114, https://dl.acm.org/doi/10.1145/3147704.3147714.
? API Patterns in C, EuroPlop 16: Proceedings of the 21st European Conference on Pattern Languages of Programs, July 2016, article no. 7, 111,https://dl.acm.org/doi/10.1145/3011784.3011791.
? Idioms for Error Handling in C, EuroPLoP 15: Proceedings of the 20th European Conference on Pattern Languages of Programs, July 2015, article no. 53, 110, https://dl.acm.org/doi/10.1145/2855321.2855377.