這是C++11領(lǐng)域最具實用性和工程實踐價值的著作。作者是金山軟件的資深工程師,有多年一線開發(fā)經(jīng)驗,致力于C++11的應用和推廣。C++11新特性眾多,作者根據(jù)自己幾年來的研究和實踐,甄選出了其中最常用和實用的部分新特性,并結(jié)合代碼實例講解了如何通過這些新特性來優(yōu)化既有的C++代碼,這部分內(nèi)容不僅能讓你迅速掌握C++11,而且能充分讓你領(lǐng)略C++11的魅力。更為重要的,作者還結(jié)合自己的企業(yè)實踐和開源項目,講解了如何利用C++11開發(fā)各種常用的工程級項目,并且所有項目的代碼均開源,具有非常高的使用價值。
全書共16章,分為兩個部分
第一部分 C++11改進我們的程序
使用類型推導、lambda、tupe元組等新特性讓程序變得更簡潔和更現(xiàn)代;
使用右值引用、move語義、emplace_back等新特性改進程序的性能;
使用可變參數(shù)模板和類型萃取等新特性消除重復和提高代碼質(zhì)量;
使用智能指針相關(guān)新特性解決內(nèi)存泄露問題;
使用線程、互斥量、異步操作等新特性讓多線程開發(fā)更簡單;
……
第二部分 C++11工程實踐
用C++11改進各種常用的設(shè)計模式,如單例模式、觀察者模式、訪問者模式,等等;
用C++11開發(fā)一個半同步半異步線程池、消息總線庫和通信程序;
用C++11開發(fā)輕量級AOP庫、輕量級IoC容器、輕量級的并行task庫
用C++11封裝SQLite庫和開發(fā)linq to objects庫;
……
為什么要寫這本書2011年C++11標準剛發(fā)布時,廣大C++開發(fā)者奔走相告,我也在第一時間看了C++之父Bjarne,Stroustrup的C++11 FAQ(http://www.stroustrup.com/C++11FAQ.html),雖然只介紹了一部分特性,而且特性的用法介紹也很簡短,但給我?guī)砣齻震撼:第一個震撼是發(fā)現(xiàn)我?guī)缀醪徽J識C++了,這么多新特性,與以前的C++很不同;第二個震撼是很多東西和其他語言類似,比如C#或者Java,感覺很酷;第三個震撼是很潮,比如lambda特性,Java都還沒有(那時Java 8還沒出來),C++11已經(jīng)有了。我是一個喜歡研究新技術(shù)的人,一下子就被C++那么多新特性吸引住了,連續(xù)幾天都在看FAQ,完全著迷了,雖然當時有很多地方?jīng)]看明白,但仍然很興奮,因為我知道這就是我想要的C++。我馬上更新編譯器嘗鮮,學習新特性。經(jīng)過一段時間的學習,在對一些主要特性有一定的了解之后,我決定在新項目中使用C++11。用C++11的感覺非常好:有了auto就不用寫冗長的類型定義了,有了lambda就不用定義函數(shù)對象了,算法也用得更舒服和自然,初始化列表讓容器和初始化變得很簡便,還有右值引用、智能指針和線程等其他很棒的特性。C++11確實讓項目的開發(fā)效率提高了很多。
相比C++98/03,C++11做了大幅度的改進,增加了相當多的現(xiàn)代編程語言的特性,使得C++的開發(fā)效率有了很大的提高。比如,C++11增加了右值引用,可以避免無謂的復制,從而提高程序性能;C++11增加了可變模板參數(shù),使C++的泛型編程能力更加強大,也大幅消除了重復模板定義;C++11增加了type_traits,可以使我們很方便地在編譯期對類型進行計算、查詢、判斷、轉(zhuǎn)換和選擇;C++11中增加的智能指針使我們不用擔心內(nèi)存泄露問題了;C++11中的線程庫讓我們能很方便地編寫可移植的并發(fā)程序。除了這些較大的改進之外,C++11還增加了很多其他實用、便利的特性,提高了開發(fā)的便利性。對于一個用過C#的開發(fā)者來說,學習C++11一定會有一種似曾相識的感覺,比如C++11的auto、for-loop循環(huán)、lambda表達式、初始化列表、tuple等分別對應了C#中的var、for-loop循環(huán)、lambda表達式、初始化列表、tuple,這些小特性使我們編寫C++程序更加簡潔和順手。C++11增加的這些特性使程序編寫變得更容易、更簡潔、更高效、更安全和更強大,那么我們還有什么理由不去學習這些特性并充分享受這些特性帶來的好處呢?學習和使用C++11不要背著C++的歷史包袱,要輕裝上陣,把它當作一門新的語言來學習,才能發(fā)現(xiàn)它的魅力和學習的樂趣。C++11增加的新特性有一百多項,很多人質(zhì)疑這會使本已復雜的C++語言變得更加復雜,從而產(chǎn)生一種抗拒心理,其實這是對C++11的誤解,C++11并沒有變得更復雜,恰恰相反,它在做簡化和改進!比如auto和decltype可以用來避免寫冗長的類型,bind綁定器讓我們不用關(guān)注到底是用bind1st還是bind2nd了,lambda表達式讓我們可以不必寫大量的不易維護的函數(shù)對象等。
語言都是在不斷進化之中的,只有跟上時代潮流的語言才是充滿活力與魅力的語言。C++正是這樣一門語言,雖然它已經(jīng)有三十多年的歷史了,但是它還在發(fā)展之中。C++14標準已經(jīng)制定完成,C++17也提上了日程,我相信C++的未來會更加美好,C++開發(fā)者的日子也會越來越美好!
作為比較早使用C++11的開發(fā)者,我開始在項目中應用C++11的時候,可以查閱的資料還很有限,主要是通過ISO標準(ISO/IEC 14882:2011)、維基百科、MSDN和http://en.cppreference.com/w/等來學習C++11。然而,這些資料對新特性的介紹比較零散,雖然知道這些新特性的基本用法,但有時候不知道為什么需要這個新特性,在實際項目中該如何應用,或者說最佳實踐是什么,這些東西網(wǎng)上可沒有,也沒有人告訴你,因為當時只有很少的人在嘗試用C++11,這些都需要自己不斷地去實踐、去琢磨,當時多么希望能有一些指導C++11實踐的資料啊。在不斷實踐的過程中,我對C++11的認識加深了,同時,也把應用C++11的一些心得和經(jīng)驗放到我的技術(shù)博客(http://www.cnblogs.com/qicosmos/)上分享出來,還開源了不少C++11的代碼,這些代碼大多來自于項目實踐。技術(shù)分享得到了很多認識的或不認識的朋友的鼓勵與支持,曾經(jīng)不止一個人問過我同一個問題,你堅持寫博客分享C++11技術(shù)是為了什么,有什么好處嗎?我想最重要的原因就是C++11讓我覺得C++語言是非常有意思和有魅力的語言,不斷給人帶來驚喜,在窺探到C++11的妙處之后,我很想和更多的人分享,讓更多的人領(lǐng)略C++11的魅力。另外一個原因是我的一點夢想,希望C++的世界變得更加美好,C++開發(fā)者的日子變得更美好。我希望這些經(jīng)驗能幫助學習C++11的朋友,讓他們少走彎路,快速地將C++11應用起來,也希望這些代碼能為使用C++的朋友帶來便利,解決他們的實際問題。
“獨樂樂,與人樂樂,孰樂乎?與少樂樂,與眾樂樂,孰樂?”,這是我分享技術(shù)和寫作此書的初衷。
讀者對象C++開發(fā)人員。
C++11新標準發(fā)布已經(jīng)4年了,C++11的使用也越來越普及,這是大勢所趨,普通的C++開發(fā)者不論是新手還是老手,都有必要學習和應用C++11,C++11強大的特性可以大幅提高生產(chǎn)率,讓我們開發(fā)項目更加得心應手。
C++11愛好者。
其他語言的開發(fā)人員,比如C#或者Java開發(fā)人員,想轉(zhuǎn)到C++開發(fā)正是時機,因為新標準的很多特性,C#和Java中也有,學起來也并不陌生,可以乘著新標準的“輕舟”學習C++11,事半功倍,正當其時。
如何閱讀本書雖然C++11的目的是為了提高生產(chǎn)率,讓C++變得更好用和更強大,但是,這些新特性畢竟很多,面對這么多特性,初學者可能會茫然無措,找不到頭緒。如果對著這些特性一一去查看標準,不僅枯燥乏味,還喪失了學習的樂趣,即使知道了新特性的基本用法,卻不知道如何應用到實際開發(fā)中。針對這兩個問題,本書試圖另辟蹊徑來解決。本書的前半部分將從另外一個角度去介紹這些新特性,不追求大而全,將重點放在一些常用的C++11特性上,有側(cè)重地從另外一個角度將這些特性分門別類,即從利用這些新特性如何去改進我們現(xiàn)有程序的角度介紹。這種方式一來可以讓讀者掌握這些新特性的用法;二來還可以讓讀者知道這些特性是如何改進現(xiàn)有程序的,從而能更深刻地領(lǐng)悟C++11的新特性。
如果說本書的前半部分貼近實戰(zhàn),那么本書后半部分的工程級應用就是真正的實戰(zhàn)。后半部分將通過豐富的開發(fā)案例來介紹如何用C++11去開發(fā)項目,因為只有在實戰(zhàn)中才能學到真東西。后半部分實戰(zhàn)案例涉及面比較廣,是筆者近年來使用C++11的經(jīng)驗與心得的總結(jié)。這些實踐經(jīng)驗是針對實際開發(fā)過程中遇到的問題來選取的,它們的價值不僅可以作為C++11實踐的指導,還可以直接在實際開發(fā)中應用(本書開發(fā)案例源碼遵循LGPL開源協(xié)議),相信這些實戰(zhàn)案例一定能給讀者帶來更深入的思考。
通過學習本書基礎(chǔ)知識與實戰(zhàn)案例,相信讀者一定能掌握大部分C++11新特性,并能應用于自己的實際開發(fā)中,充分享受C++11帶來的好處。
C++之父BjarneStroustrup曾說過:C++11看起來像一門新的語言。這個說法是否夸張,讀者不妨看完本書之后再來回味這句話。
祁 宇 資深C++技術(shù)專家,致力于C++11的應用、研究和推廣。金山軟件WPS資深工程師,負責Android服務(wù)端開發(fā)。精通OOP、OOD、設(shè)計模式和重構(gòu),主要研究方向為架構(gòu)設(shè)計和業(yè)務(wù)重構(gòu),有豐富的開發(fā)和研發(fā)管理經(jīng)驗。愛好C++,愛好開源,樂于研究和分享技術(shù),開源了多個項目,在《程序員》雜志發(fā)表多篇技術(shù)文章。
2013年被評為珠海市優(yōu)秀青年人才。
前言
第一篇 C++11改進我們的程序
第1章 使用C++11讓程序更簡潔、更現(xiàn)代,
1.1 類型推導
1.1.1 auto類型推導
1.1.2 decltype關(guān)鍵字
1.1.3 返回類型后置語法——auto和decltype的結(jié)合使用
1.2 模板的細節(jié)改進
1.2.1 模板的右尖括號
1.2.2 模板的別名
1.2.3 函數(shù)模板的默認模板參數(shù)
1.3 列表初始化
1.3.1 統(tǒng)一的初始化
1.3.2 列表初始化的使用細節(jié)
1.3.3 初始化列表
1.3.4 防止類型收窄
1.4 基于范圍的for循環(huán)
1.4.1 for循環(huán)的新用法
1.4.2 基于范圍的 for循環(huán)的使用細節(jié)
1.4.3 讓基于范圍的 for循環(huán)支持自定義類型
1.5 std::function和bind綁定器
1.5.1 可調(diào)用對象
1.5.2 可調(diào)用對象包裝器——std::function
1.5.3 std::bind綁定器
1.6 lambda表達式
1.6.1 lambda表達式的概念和基本用法
1.6.2 聲明式的編程風格,簡潔的代碼
1.6.3 在需要的時間和地點實現(xiàn)閉包,使程序更靈活
1.7 tupe元組
1.8 總結(jié)
第2章 使用C++11改進程序性能
2.1 右值引用
2.1.1 &&的特性
2.1.2 右值引用優(yōu)化性能,避免深拷貝
2.2 move語義
2.3 forward和完美轉(zhuǎn)發(fā)
2.4 emplace_back減少內(nèi)存拷貝和移動
2.5 unordered container無序容器
2.6 總結(jié)
第3章 使用C++11消除重復,提高代碼質(zhì)量
3.1 type_traits——類型萃取
3.1.1 基本的type_traits
3.1.2 根據(jù)條件選擇的traits
3.1.3 獲取可調(diào)用對象返回類型的traits
3.1.4 根據(jù)條件禁用或啟用某種或某些類型traits
3.2 可變參數(shù)模板
3.2.1 可變參數(shù)模板函數(shù)
3.2.2 可變參數(shù)模板類
3.2.3 可變參數(shù)模板消除重復代碼
3.3 可變參數(shù)模版和type_taits的綜合應用
3.3.1 optional的實現(xiàn)
3.3.2 惰性求值類lazy的實現(xiàn)
3.3.3 dll幫助類
3.3.4 lambda鏈式調(diào)用
3.3.5 any類的實現(xiàn)
3.3.6 function_traits
3.3.7 variant的實現(xiàn)
3.3.8 ScopeGuard
3.3.9 tuple_helper
3.4 總結(jié)
第4章 使用C++11解決內(nèi)存泄露的問題
4.1 shared_ptr共享的智能指針
4.1.1 shared_ptr的基本用法
4.1.2 使用shared_ptr需要注意的問題
4.2 unique_ptr獨占的智能指針
4.3 weak_ptr弱引用的智能指針
4.3.1 weak_ptr基本用法
4.3.2 weak_ptr返回this指針
4.3.3 weak_ptr解決循環(huán)引用問題
4.4 通過智能指針管理第三方庫分配的內(nèi)存
4.5 總結(jié)
第5章 使用C++11讓多線程開發(fā)變得簡單
5.1 線程
5.1.1 線程的創(chuàng)建
5.1.2 線程的基本用法
5.2 互斥量
5.2.1 獨占互斥量std::mutex
5.2.2 遞歸互斥量std::recursive_mutex
5.2.3 帶超時的互斥量std::timed_mutex和std::recursive_timed_mutex
5.3 條件變量
5.4 原子變量
5.5 call_once/once_flag的使用
5.6 異步操作
5.6.1 獲取線程函數(shù)返回值的類std::future
5.6.2 協(xié)助線程賦值的類 std::promise
5.6.3 可調(diào)用對象的包裝類std::package_task
5.6.4 std::promise、std::packaged_task和std::future三者之間的關(guān)系
5.7 線程異步操作函數(shù)async
5.8 總結(jié)
第6章 使用C++11中便利的工具
6.1 處理日期和時間的chrono庫
6.1.1 記錄時長的duration
6.1.2 表示時間點的time point
6.1.3 獲取系統(tǒng)時鐘的clocks
6.1.4 計時器timer
6.2 數(shù)值類型和字符串的相互轉(zhuǎn)換
6.3 寬窄字符轉(zhuǎn)換
6.4 總結(jié)
第7章 C++11的其他特性
7.1 委托構(gòu)造函數(shù)和繼承構(gòu)造函數(shù)
7.1.1 委托構(gòu)造函數(shù)
7.1.2 繼承構(gòu)造函數(shù)
7.2 原始的字面量
7.3 f?inal和override關(guān)鍵字
7.4 內(nèi)存對齊
7.4.1 內(nèi)存對齊介紹
7.4.2 堆內(nèi)存的內(nèi)存對齊
7.4.3 利用alignas指定內(nèi)存對齊大小
7.4.4 利用alignof和std::alignment_of獲取內(nèi)存對齊大小
7.4.5 內(nèi)存對齊的類型std::aligned_storage
7.4.6 std::max_align_t和std::align操作符
7.5 C++11新增的便利算法
7.6 總結(jié)
第二篇 C++11工程級應用
第8章 使用C++11改進我們的模式
8.1 改進單例模式
8.2 改進觀察者模式
8.3 改進訪問者模式
8.4 改進命令模式
8.5 改進對象池模式
8.6 總結(jié)
第9章 使用C++11開發(fā)一個半同步半異步線程池
9.1 半同步半異步線程池介紹
9.2 線程池實現(xiàn)的關(guān)鍵技術(shù)分析
9.3 同步隊列
9.4 線程池
9.5 應用實例
9.6 總結(jié)
第10章 使用C++11開發(fā)一個輕量級的AOP庫
10.1 AOP介紹
10.2 AOP的簡單實現(xiàn)
10.3 輕量級的AOP框架的實現(xiàn)
10.4 總結(jié)
第11章 使用C++11開發(fā)一個輕量級的IoC容器
11.1 IoC容器是什么
11.2 IoC創(chuàng)建對象
11.3 類型擦除的常用方法
11.4 通過Any和閉包來擦除類型
11.5 創(chuàng)建依賴的對象
11.6 完整的IoC容器
11.7 總結(jié)
第12章 使用C++11開發(fā)一個對象的消息總線庫
12.1 消息總線介紹
12.2 消息總線關(guān)鍵技術(shù)
12.2.1 通用的消息定義
12.2.2 消息的注冊
12.2.3 消息分發(fā)
12.2.4 消息總線的設(shè)計思想
12.3 完整的消息總線
12.4 應用實例
12.5 總結(jié)
第13章 使用C++11封裝sqlite庫
13.1 sqlite基本用法介紹
13.1.1 打開和關(guān)閉數(shù)據(jù)庫的函數(shù)
13.1.2 執(zhí)行SQL語句的函數(shù)
13.2 rapidjson基本用法介紹
13.2.1 解析json字符串
13.2.2 創(chuàng)建json對象
13.2.3 對rapidjson的一點擴展
13.3 封裝sqlite的SmartDB
13.3.1 打開和關(guān)閉數(shù)據(jù)庫的接口
13.3.2 Excecute接口
13.3.3 ExecuteScalar接口
13.3.4 事務(wù)接口
13.3.5 ExcecuteTuple接口
13.3.6 json接口
13.3.7 查詢接口
13.4 應用實例
13.5 總結(jié)
第14章 使用C++11開發(fā)一個linq to objects庫
14.1 LINQ介紹
14.1.1 LINQ語義
14.1.2 Linq標準操作符(C#)
14.2 C++中的LINQ
14.3 LINQ實現(xiàn)的關(guān)鍵技術(shù)
14.3.1 容器和數(shù)組的泛化
14.3.2 支持所有的可調(diào)用對象
14.3.3 鏈式調(diào)用
14.4 linq to objects的具體實現(xiàn)
14.4.1 一些典型LINQ操作符的實現(xiàn)
14.4.2 完整的linq to objects的實現(xiàn)
14.5 linq to objects的應用實例
14.6 總結(jié)
第15章 使用C++11開發(fā)一個輕量級的并行task庫
15.1 TBB的基本用法
15.1.1 TBB概述
15.1.2 TBB并行算法
15.1.3 TBB的任務(wù)組
15.2 PPL的基本用法
15.2.1 PPL任務(wù)的鏈式連續(xù)執(zhí)行
15.2.2 PPL的任務(wù)組
15.3 TBB和PPL的選擇
15.4 輕量級的并行庫TaskCpp的需求
15.5 TaskCpp的任務(wù)
15.5.1 task的實現(xiàn)
15.5.2 task的延續(xù)
15.6 TaskCpp任務(wù)的組合
15.6.1 TaskGroup
15.6.2 WhenAll
15.6.3 WhenAny
15.7 TaskCpp并行算法
15.7.1 ParallelForeach:并行對區(qū)間元素執(zhí)行某種操作
15.7.2 ParallelInvoke:并行調(diào)用
15.7.3 ParallelReduce:并行匯聚
15.8 總結(jié)
第16章 使用C++11開發(fā)一個簡單的通信程序
16.1 反應器和主動器模式介紹
16.2 asio中的Proactor
16.3 asio的基本用法
16.3.1 異步接口
16.3.2 異步發(fā)送
16.4 C++11結(jié)合asio實現(xiàn)一個簡單的服務(wù)端程序
16.5 C++11結(jié)合asio實現(xiàn)一個簡單的客戶端程序
16.6 TCP粘包問題的解決
16.7 總結(jié)
參考文獻