日韩av黄I国产麻豆传媒I国产91av视频在线观看I日韩一区二区三区在线看I美女国产在线I麻豆视频国产在线观看I成人黄色短片

歡迎訪問 生活随笔!

生活随笔

當(dāng)前位置: 首頁 > 编程语言 > c/c++ >内容正文

c/c++

编写高质量代码改善C++程序的150个建议

發(fā)布時(shí)間:2023/12/29 c/c++ 55 豆豆
生活随笔 收集整理的這篇文章主要介紹了 编写高质量代码改善C++程序的150个建议 小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,幫大家做個(gè)參考.

第一部分 語法篇

?

第1章???從C繼承而來的

建議0:不用讓main函數(shù)返回void

???????? main函數(shù)的返回類型是int,不是void或其它類型。

建議1:區(qū)分0的4種面孔

?????????(1)、整型0;(2)、空指針NULL,指針與int類型所占空間是一樣的,都是32位;(3)、字符串結(jié)束標(biāo)志’\0’;(4)、邏輯FALSE/false,FALSE/TRUE是int類型,而false/true是bool類型。

建議2:避免那些由運(yùn)算符引發(fā)的混亂

???????? 不要混淆=和==、&和&&、|與||這三對(duì)運(yùn)算符之間的差別,針對(duì)=和==之間的問題,可以這樣做:if (0 == nValue)。

建議3:對(duì)表達(dá)式計(jì)算順序不用想當(dāng)然

?????????(1)、針對(duì)操作符優(yōu)先級(jí),建議多寫幾個(gè)括號(hào);(2)、注意函數(shù)參數(shù)和操作數(shù)的評(píng)估求值順序問題。

建議4:小心宏#define使用中的陷阱

?????????(1)、用宏定義表達(dá)式時(shí),要使用完備的括號(hào):如 #define ADD(a, b) ((a)+(b));(2)、使用宏時(shí),不允許參數(shù)發(fā)生變化;(3)、用大括號(hào)將宏所定義的多條表達(dá)式括起來。

建議5:不要忘記指針變量的初始化

?????????(1)、可以將其初始化為空指針0(NULL);(2)、對(duì)于全局變量來說,在聲明的同時(shí),編譯器會(huì)悄悄完成對(duì)變量的初始化。

建議6:明晰逗號(hào)分隔表達(dá)式的奇怪之處

?????????(1)、在使用逗號(hào)分隔表達(dá)式時(shí),C++會(huì)確保每個(gè)表達(dá)式都被執(zhí)行,而整個(gè)表達(dá)式的值則是最右邊表達(dá)式的結(jié)果;(2)、在C++中,逗號(hào)分隔表達(dá)式既可以用作左值,也可以用作右值。

建議7:時(shí)刻提防內(nèi)存溢出

?????????在調(diào)用C語言字符串金典函數(shù)(strcpy、strcat、gets等)時(shí),要從源代碼開始就提高警惕,盡量追蹤傳入數(shù)據(jù)的流向,向代碼中的每一個(gè)假設(shè)提出質(zhì)疑。在訪問數(shù)據(jù)時(shí),注意對(duì)于邊界數(shù)據(jù)要特殊情況特殊處理,還要對(duì)杜絕使用未初始化指針和失敗后未置NULL的“野指針”。

建議8:拒絕晦澀難懂的函數(shù)指針

???????? 函數(shù)指針在運(yùn)行時(shí)的動(dòng)態(tài)調(diào)用(例如函數(shù)回調(diào))中應(yīng)用廣泛。但是直接定義復(fù)雜的函數(shù)指針會(huì)由于有太多的括號(hào)而使代碼的可讀性下降。使用typedef可以讓函數(shù)指針更直觀和易維護(hù)。

建議9:防止重復(fù)包含頭文件

???????? 為了避免重復(fù)包含頭文件,建議在聲明每個(gè)頭文件時(shí)采用“頭文件衛(wèi)士”加以保護(hù),比如采用如下的形式:

#ifndef__SOMEFILE_H__#define__SOMEFILE_H__……//聲明、定義語句#endif

建議10:優(yōu)化結(jié)構(gòu)體中元素的布局

?????????把結(jié)構(gòu)體中的變量按照類型大小從小到大依次聲明,盡量減少中間的填充字節(jié)。

建議11:將強(qiáng)制轉(zhuǎn)型減到最少

???????? (1)、const_cast<T*>(a):它用于從一個(gè)類中去除以下這些屬性:const、volatile和__unaligned;(2)、dynamic_cast<T*>(a):它將a值轉(zhuǎn)換成類型為T的對(duì)象指針,主要用來實(shí)現(xiàn)類層次結(jié)構(gòu)的提升;(3)、reinterpret_cast<T*>(a):它能夠用于諸如One_class*到Unrelated_class*這樣的不相關(guān)類型之間的轉(zhuǎn)換,因此它是不安全的;(4)、static_cast<T*>(a):它將a的值轉(zhuǎn)換為模板中指定的類型T,但是,在運(yùn)行時(shí)轉(zhuǎn)換過程中,它不會(huì)進(jìn)行類型檢查,不能確保轉(zhuǎn)換的安全性。

建議12:優(yōu)先使用前綴操作符

?????????對(duì)于整型和長整型的操作,前綴操作和后綴操作的性能區(qū)別通常是可以忽略的。對(duì)于用戶自定義類型,優(yōu)先使用前綴操作符。因?yàn)榕c后綴操作符相比,前綴操作符因?yàn)闊o須構(gòu)造臨時(shí)對(duì)象而更具性能優(yōu)勢。

建議13:掌握變量定義的位置與時(shí)機(jī)

?????????在定義變量時(shí),要三思而后行,掌握變量定義的時(shí)機(jī)與位置,在合適的時(shí)機(jī)于合適的位置上定義變量。盡可能推遲變量的定義,直到不得不需要該變量為止;同時(shí),為了減少變量名污染,提高程序的可讀性,盡量縮小變量的作用域。

建議14:小心typedef使用中的陷阱

???????? 區(qū)分typedef與#define之間的不同;不要用理解宏的思維方式對(duì)待typedef,typedef聲明的新名稱具有一定的封裝性,更易定義變量。同時(shí)還要注意它是一個(gè)無“現(xiàn)實(shí)意義”的存儲(chǔ)類關(guān)鍵字。

建議15:盡量不要使用可變參數(shù)

?????????編譯器對(duì)可變參數(shù)函數(shù)的原型檢查不夠嚴(yán)格,所以容易引起問題,難于查錯(cuò),不利于寫出高質(zhì)量的代碼。所以應(yīng)當(dāng)盡量避免使用C語言方式的可變參數(shù)設(shè)計(jì),而用C++中更為安全的方式來完美代替之(如多態(tài)等)。

建議16:慎用goto

?????????過度使用goto會(huì)使代碼流程錯(cuò)綜復(fù)雜,難以理清頭緒。所以,如果不熟悉goto,不要使用它;如果已經(jīng)習(xí)慣使用它,試著不去使用。

建議17:提防隱式轉(zhuǎn)換帶來的麻煩

?????????提防隱式轉(zhuǎn)換所帶來的微妙問題,盡量控制隱式轉(zhuǎn)換的發(fā)生;通常采用的方式包括:(1)、使用非C/C++關(guān)鍵字的具名函數(shù),用operator as_T()替換operator T()(T為C++數(shù)據(jù)類型)。(2)、為單參數(shù)的構(gòu)造函數(shù)加上explicit關(guān)鍵字。

建議18:正確區(qū)分void與void*

?????????Void是“無類型”,所以它不是一種數(shù)據(jù)類型;void*則為“無類型指針”,即它是指向無類型數(shù)據(jù)的指針,也就是說它可以指向任何類型的數(shù)據(jù)。Void發(fā)揮的真正作用是限制程序的參數(shù)與函數(shù)返回值:(1)、如果函數(shù)沒有返回值,那么應(yīng)將其聲明為void類型;(2)、如果函數(shù)無參數(shù),那么聲明函數(shù)參數(shù)為void。對(duì)于void*,(1)、任何類型的指針都可以直接賦值給它,無須強(qiáng)制轉(zhuǎn)型;(2)、如果函數(shù)的參數(shù)可以是任意類型指針,那么應(yīng)聲明其參數(shù)為void*。

?

第2章???從C到C++,需要做出一些改變

建議19:明白在C++中如何使用C

???????? 若想在C++中使用大量現(xiàn)成的C程序庫,就必須把它放到extern“C” {/* code */}中,extern “C”的作用就是告訴C++鏈接器尋找調(diào)用函數(shù)的符號(hào)時(shí),采用C的方式。要實(shí)現(xiàn)在C++代碼中調(diào)用C的代碼,具體方式有以下幾種:(1)、修改C代碼的頭文件,當(dāng)其中含有C++代碼時(shí),在聲明中加入extern “C”;(2)、在C++代碼中重新聲明一下C函數(shù),在重新聲明時(shí)添加上extern “C”;(3)、在包含C頭文件時(shí),添上extern “C”。

建議20:使用memcpy()系列函數(shù)時(shí)要足夠小心

?????????要區(qū)分哪些數(shù)據(jù)對(duì)象是POD(傳統(tǒng)C風(fēng)格的數(shù)據(jù)類型,C的所有對(duì)象都是POD,對(duì)于任何POD對(duì)象,我們都可以放心大膽地使用memset()、memcpy()、memcmp()等函數(shù)對(duì)對(duì)象的內(nèi)存數(shù)據(jù)進(jìn)行操作),哪些是非POD(C++的對(duì)象可能并不是一個(gè)POD,如動(dòng)多態(tài)),由于非POD對(duì)象的存在,在C++中使用memcpy()系列函數(shù)時(shí)要保持足夠的小心。

建議21:盡量用new/delete代替malloc/free

?????????malloc與new之間的區(qū)別:(1)、new是C++運(yùn)算符,而malloc則是C標(biāo)準(zhǔn)庫函數(shù);(2)、通過new創(chuàng)建的東西是具有類型的,而malloc函數(shù)返回的則是void*,需要進(jìn)行強(qiáng)制轉(zhuǎn)型;(3)、new可以自動(dòng)調(diào)用對(duì)象的構(gòu)造函數(shù),而malloc不會(huì);(4)、new失敗是會(huì)調(diào)用new_handler處理函數(shù),而malloc失敗則直接返回NULL。Free與delete之間的區(qū)別:(1)、delete是C++運(yùn)算符,free是C標(biāo)準(zhǔn)庫函數(shù);(2)、delete可以自動(dòng)調(diào)用對(duì)象的析構(gòu)函數(shù),而malloc不會(huì)。另外,new/delete必須配對(duì)使用,malloc/free也一樣。

建議22:靈活地使用不同風(fēng)格的注釋

?????????C風(fēng)格的注釋/* */與C++風(fēng)格的注釋//在C++語言中同時(shí)存在。

建議23:盡量使用C++標(biāo)準(zhǔn)的iostream

?????????建議使用#include <iostream>

建議24:盡量采用C++風(fēng)格的強(qiáng)制轉(zhuǎn)型

建議25:盡量用const、enum、inline替換#define

?????????對(duì)于簡單的常量,應(yīng)該盡量使用const對(duì)象或枚舉類型數(shù)據(jù),避免使用#define;對(duì)于形似函數(shù)的宏,盡量使用內(nèi)聯(lián)函數(shù),避免使用#define。總之一句話,盡量將工作交給編譯器,而不是預(yù)處理器。

建議26:用引用代替指針

?????????與指針不同,引用與地址沒有關(guān)聯(lián),甚至不占任何存儲(chǔ)空間。

?

第3章???說一說“內(nèi)存管理”的那點(diǎn)事兒

在VC中,棧空間未初始化的字符默認(rèn)是-52,補(bǔ)碼是0xCC,兩個(gè)0xCC,即0xCCCC在GBK編碼中就是“燙”;堆空間未初始化的字符默認(rèn)是-51,兩個(gè)-51在GBK編碼中就是“屯”。兩者都是未初始化的內(nèi)存。

?

建議27:區(qū)分內(nèi)存分配的方式

?????????一個(gè)程序要運(yùn)行,就必須先將可執(zhí)行的程序加載到計(jì)算機(jī)內(nèi)存里,程序加載完畢后,就可以形成一個(gè)運(yùn)行空間,按照代碼區(qū)、數(shù)據(jù)區(qū)、堆區(qū)、棧區(qū)進(jìn)行布局。代碼區(qū)存放的是程序的執(zhí)行代碼;數(shù)據(jù)區(qū)存放的是全局?jǐn)?shù)據(jù)、常量、靜態(tài)變量等;堆區(qū)存放的則是動(dòng)態(tài)內(nèi)存,供程序隨機(jī)申請(qǐng)使用;而棧區(qū)則存放著程序中所用到的局部數(shù)據(jù)。這些數(shù)據(jù)可以動(dòng)態(tài)地反應(yīng)程序中對(duì)函數(shù)的調(diào)用狀態(tài),通過其軌跡也可以研究其函數(shù)機(jī)制。

???????? 在C++中,數(shù)據(jù)區(qū)又被分成自由存儲(chǔ)區(qū)、全局/靜態(tài)存儲(chǔ)區(qū)和常量存儲(chǔ)區(qū),再加上堆區(qū)、棧區(qū),也就是說,內(nèi)存被分成了5個(gè)區(qū)。(1)、棧區(qū):在執(zhí)行函數(shù)時(shí),函數(shù)內(nèi)局部變量的存儲(chǔ)單元都可以在棧上創(chuàng)建,函數(shù)執(zhí)行結(jié)束時(shí)這些存儲(chǔ)單元將自動(dòng)被釋放。棧內(nèi)存分配運(yùn)算內(nèi)置于處理器的指令集中,效率很高,但是所分配的內(nèi)存容量有限;(2)、堆區(qū):堆就是那些由new分配的內(nèi)存塊,其釋放編譯器不會(huì)管它,而是由我們的應(yīng)用程序控制它,一般一個(gè)new就對(duì)應(yīng)于一個(gè)delete,如果程序員沒有釋放掉,那么在程序結(jié)束后,操作系統(tǒng)就會(huì)自動(dòng)回收;(3)、自由存儲(chǔ)區(qū):是那些由malloc等分配的內(nèi)存塊,它和堆十分相似,不過它是用free來結(jié)束自己生命的;(4)、全局/靜態(tài)存儲(chǔ)區(qū):全局變量和靜態(tài)變量被分配到同一塊內(nèi)存中,在以前的C語言中,全局變量又分為初始化的和未初始化的,在C++里面沒有作此區(qū)分,它們共同占用同一塊內(nèi)存區(qū);(5)、常量存儲(chǔ)區(qū):這是一塊比較特殊的存儲(chǔ)區(qū),里面存放的是常量,不允許修改。

???????? 堆與棧的區(qū)別:(1)、管理方式不同:對(duì)于棧來講,它是由編譯器自動(dòng)管理的,無須我們手工控制;對(duì)于堆來說,它的釋放工作由程序員控制,容易產(chǎn)生memory leak;(2)、空間大小不同:一般來講在32位系統(tǒng)下,堆內(nèi)存可以達(dá)到4GB的空間,從這個(gè)角度來看堆內(nèi)存幾乎是沒有什么限制的。但是對(duì)于棧來講,一般都是有一定空間大小的;(3)、碎片問題:對(duì)于堆來講,頻繁的new/delete勢必會(huì)造成內(nèi)存空間的不連續(xù),從而產(chǎn)生大量的碎片,使程序效率降低。對(duì)于棧來講,則不存在這個(gè)問題,其原因還要從棧的特殊數(shù)據(jù)結(jié)構(gòu)說起。棧是一個(gè)具有嚴(yán)明紀(jì)律的隊(duì)列,其中的數(shù)據(jù)必須遵循先進(jìn)后出的規(guī)則,相互之間緊密排列,絕不會(huì)留給其他數(shù)據(jù)可插入之空隙,所以永遠(yuǎn)都不可能有一個(gè)內(nèi)存塊從棧中間彈出,它們必須嚴(yán)格按照一定的順序一一彈出;(4)、生成方向:對(duì)于堆來講,其生長方向是向上的,也就是向著內(nèi)存地址增加的方向增長;對(duì)于棧來講,它的生長方向是向下的,是向著內(nèi)存地址減小的方向增長的;(5)、分配方式:堆都是動(dòng)態(tài)分配的,沒有靜態(tài)分配的堆。棧有兩種分配方式:靜態(tài)分配和動(dòng)態(tài)分配。靜態(tài)分配是編譯器完成的,比如局部變量的分配。動(dòng)態(tài)分配由alloca函數(shù)完成,但是棧的動(dòng)態(tài)分配和堆是不同的,它的動(dòng)態(tài)分配是由編譯器進(jìn)行釋放的,無須我們手工實(shí)現(xiàn);(6)、分配效率:棧是機(jī)器系統(tǒng)提供的數(shù)據(jù)結(jié)構(gòu),計(jì)算機(jī)會(huì)在底層對(duì)棧提供支持:它會(huì)分配專門的寄存器存放棧的地址,而且壓棧出棧都會(huì)有專門的指令來執(zhí)行,這就決定了棧的效率比較高。堆則是C/C++函數(shù)庫提供的,它的機(jī)制很復(fù)雜,例如為了分配一塊內(nèi)存,庫函數(shù)會(huì)按照一定的算法在堆內(nèi)存中搜索可用的足夠大小的空間,如果沒有足夠大小的空間(可能是由于內(nèi)存碎片太多),則可能調(diào)用系統(tǒng)功能去增加程序數(shù)據(jù)段的內(nèi)存空間,這樣就有機(jī)會(huì)分到足夠大小的內(nèi)存了,然后返回。顯然,堆的效率比棧要低得多。

建議28:new/delete與new[]/delete[]必須配對(duì)使用

?????????由于內(nèi)置數(shù)據(jù)類型沒有構(gòu)造、析構(gòu)函數(shù),所以在針對(duì)內(nèi)置數(shù)據(jù)類型時(shí),釋放內(nèi)存使用delete或delete[]的效果都是一樣的。

建議29:區(qū)分new的三種形態(tài)

???????? (1)、如果是在堆上建立對(duì)象,那么應(yīng)該使用new operator,它會(huì)為你提供最為周全的服務(wù);(2)、如果僅僅是分配內(nèi)存,那么應(yīng)該調(diào)用operator new,但初始化不在它的工作職責(zé)之內(nèi)。如果你對(duì)默認(rèn)的內(nèi)存分配過程不滿意,想單獨(dú)定制,重載operator new是不二選擇;(3)、如果想在一塊已經(jīng)獲得的內(nèi)存里建立一個(gè)對(duì)象,那就應(yīng)該用placement new。但是通常情況下不建議使用,除非是在某些對(duì)時(shí)間要求非常高的應(yīng)用中,因?yàn)橄鄬?duì)于其他兩個(gè)步驟,選擇合適的構(gòu)造函數(shù)完成對(duì)象初始化是一個(gè)時(shí)間相對(duì)較長的過程。

建議30:new內(nèi)存失敗后的正確處理

?????????當(dāng)使用new申請(qǐng)一塊內(nèi)存失敗時(shí),拋出異常std::bad_alloc是C++標(biāo)準(zhǔn)中規(guī)定的標(biāo)準(zhǔn)行為,所以推薦使用try{p=new int[SIZE];} catch(std::bad_alloc) {…} 的處理方式。但是在一些老舊的編譯器中,卻不支持該標(biāo)準(zhǔn),它會(huì)返回NULL,此時(shí)具有C傳統(tǒng)的Test_for_NULL代碼形式便起了作用。所以,要針對(duì)不同的情形采取合理的處置方式。

建議31:了解new_handler的所作所為

???????? 在使用operatornew申請(qǐng)內(nèi)存失敗后,編譯器并不是不做任何的努力直接拋出std::alloc異常,在這之前,它會(huì)調(diào)用一個(gè)錯(cuò)誤處理函數(shù)(這個(gè)函數(shù)被稱為new-handler),進(jìn)行相應(yīng)的處理。通常,一個(gè)好的new-handler函數(shù)的處理方式必須遵循以下策略之一:(1)、使更大塊內(nèi)存有效;(2)、裝載另外的new-handler;(3)、卸載new-handler;(4)、拋出異常;(5)、無返回。

建議32:借助工具檢測內(nèi)存泄露問題

?????????內(nèi)存泄露一般指的是堆內(nèi)存的泄露。檢測內(nèi)存泄露的關(guān)鍵是能截獲對(duì)分配內(nèi)存和釋放內(nèi)存的函數(shù)的調(diào)用。通過截獲的這兩個(gè)函數(shù),我們就能跟蹤每一塊內(nèi)存的生命周期。每當(dāng)成功分配一塊內(nèi)存時(shí),就把它的指針加入一個(gè)全局的內(nèi)存鏈中;每當(dāng)釋放一塊內(nèi)存時(shí),再把它的指針從內(nèi)存鏈中刪除。這樣當(dāng)程序運(yùn)行結(jié)束的時(shí)候,內(nèi)存鏈中剩余的指針就會(huì)指向那些沒有被釋放的內(nèi)存。這就是檢測內(nèi)存泄露的基本原理。

???????? 檢測內(nèi)存泄露的常用方法有如下幾種:(1)、MS C-Runtime Library內(nèi)建的檢測功能,要在非MFC程序中打開內(nèi)存泄露的檢測功能非常容易,只須在程序的入口處添加以下代碼:

_CrtSetDbgFlag(_CrtSetDbgFlag(_CRTDBG_REPORT_FLAG)| _CRTDBG_LEAK_CHECK_DF);

(2)、外掛式的檢測工具:MS下BoundsChecker或Insure++;Linux下RationalPurify或Valgrind.

建議33:小心翼翼地重載operator new/operator delete

?????????通過重載operator new 和 operatordelete的方法,可以自由地采用不同的分配策略,從不同的內(nèi)存池中分配不同的類對(duì)象。但是是否選擇重載operator new/delete 一定要深思熟慮。

建議34:用智能指針管理通過new創(chuàng)建的對(duì)象

?????????智能指針auto_ptr,要使用它,需要包含memory頭文件:(1)、auto_ptr對(duì)象不可作為STL容器的元素;(2)、auto_ptr缺少對(duì)動(dòng)態(tài)配置而來的數(shù)組的支持;(3)、auto_ptr在被復(fù)制的時(shí)候會(huì)發(fā)生所有權(quán)轉(zhuǎn)移。

建議35:使用內(nèi)存池技術(shù)提高內(nèi)存申請(qǐng)效率與性能

?????????經(jīng)典的內(nèi)存池技術(shù),是一種用于分配大量大小相同的小對(duì)象的技術(shù)。通過該技術(shù)可以極大地加快內(nèi)存分配/釋放過程。內(nèi)存池技術(shù)通過批量申請(qǐng)內(nèi)存,降低了內(nèi)存申請(qǐng)次數(shù),從而節(jié)省了時(shí)間。

?

第4章???重中之重的類

建議36:明晰class與struct之間的區(qū)別

?????????C中的struct是一種數(shù)據(jù)類型。C++中struct被看成是一個(gè)對(duì)象,它可以包含函數(shù),可以擁有構(gòu)造函數(shù)、析構(gòu)函數(shù),同樣擁有繼承等能力。

???????? C++中的class與struct差別:(1)、class和struct如果定義了構(gòu)造函數(shù),就都不能用大括號(hào)進(jìn)行初始化了;如果沒有定義,struct可以用大括號(hào)初始化,而class只有在所有成員變量全是public的情況下,才可以用大括號(hào)進(jìn)行初始化;(2)、關(guān)于默認(rèn)訪問權(quán)限:class中默認(rèn)的成員訪問權(quán)限是private的,而struct中則是public的;(3)、關(guān)于繼承方式:class繼承默認(rèn)是private繼承,而struct默認(rèn)是public繼承

建議37:了解C++悄悄做的那些事

???????? 對(duì)于類,編譯器會(huì)悄悄地完成很多事:隱士產(chǎn)生一個(gè)類的默認(rèn)構(gòu)造函數(shù),拷貝構(gòu)造函數(shù),拷貝賦值運(yùn)算符和析構(gòu)函數(shù)。

建議38:首選初始化列表實(shí)現(xiàn)類成員的初始化

?????????類成員的初始化可采用兩種形式來完成:在構(gòu)造函數(shù)體重賦值完成和用初始化類成員列表完成:(1)、const成員變量只能用成員初始化列表來完成初始化,而不能在構(gòu)造函數(shù)內(nèi)被賦值;(2)、如果類B中含有A類型的成員變量,而類A中又禁止了賦值操作,此時(shí)要想順利地完成B中成員變量的初始化,就必須采用初始化列表方式。即使沒有禁用賦值操作,還是不推薦采用函數(shù)體內(nèi)的賦值初始化方式。因?yàn)檫@種方式存在著兩種問題。第一,比起初始化列表,此方式效率偏低;第二,留有錯(cuò)誤隱患。

???????? 對(duì)于初始化列表,初始化的順序與構(gòu)造函數(shù)中的賦值方式不同,初始化列表中成員變量出現(xiàn)的順序并不是真正初始化的順序,初始化的順序取決于成員變量在類中的聲明順序。只有保證成員變量聲明的順序與初始化列表順序一致才能真正保證其效率。

建議39:明智地拒絕對(duì)象的復(fù)制操作

?????????在某些需要禁止對(duì)象復(fù)制操作的情形下,可以將這個(gè)類相應(yīng)的拷貝構(gòu)造函數(shù)、賦值操作符operator = 聲明為private,并且不要給出實(shí)現(xiàn)。或者采用更簡單的方法:使用boost::noncopyable作為基類。

建議40:小心,自定義拷貝函數(shù)

?????????如果類內(nèi)部出現(xiàn)了動(dòng)態(tài)配置的資源,我們就不得不自定義實(shí)現(xiàn)其拷貝函數(shù)了。在自定義拷貝函數(shù)時(shí),應(yīng)該保證拷貝一個(gè)對(duì)象的All Parts:所有數(shù)據(jù)成員及所有的基類部分。

建議41:謹(jǐn)防因構(gòu)造函數(shù)拋出異常而引發(fā)的問題

?????????判斷構(gòu)造對(duì)象成功與否,解決辦法:拋出一個(gè)異常。構(gòu)造函數(shù)拋出異常會(huì)引起對(duì)象的部分構(gòu)造,因?yàn)椴荒茏詣?dòng)調(diào)用析構(gòu)函數(shù),在異常發(fā)生之前分配的資源將得不到及時(shí)的清理,進(jìn)而造成內(nèi)存泄露問題。所以,如果對(duì)象中涉及了資源分配,一定要對(duì)構(gòu)造之中可能拋出的異常做謹(jǐn)慎而細(xì)致的處理。

建議42:多態(tài)基類的析構(gòu)函數(shù)應(yīng)該為虛

?????????虛函數(shù)的最大目的就是允許派生類定制實(shí)現(xiàn)。所以,用基類指針刪除一個(gè)派生類對(duì)象時(shí),C++會(huì)正確地調(diào)用整個(gè)析構(gòu)鏈,執(zhí)行正確的行為,以銷毀整個(gè)對(duì)象。在實(shí)際使用虛析構(gòu)函數(shù)的過程中,一般要遵守以下規(guī)則:當(dāng)類中包含至少一個(gè)虛函數(shù)時(shí),才將該類的析構(gòu)函數(shù)聲明為虛。因?yàn)橐粋€(gè)類要作為多態(tài)基類使用時(shí),它一定會(huì)包含一個(gè)需要派生定制的虛函數(shù)。相反,如果一個(gè)類不包含虛函數(shù),那就預(yù)示著這個(gè)類不能作為多態(tài)基類使用。同樣,如果一個(gè)類的析構(gòu)函數(shù)非虛,那你就要頂住誘惑,決不能繼承它,即使它是“出身名門”。比如標(biāo)準(zhǔn)庫中的string、complex、以及STL容器。

???????? 多態(tài)基類的析構(gòu)函數(shù)應(yīng)該是virtual的,也必須是virtual的,因?yàn)橹挥羞@樣,虛函數(shù)機(jī)制才會(huì)保證派生類對(duì)象的徹底釋放;如果一個(gè)類有一個(gè)虛函數(shù),那么它就該有一個(gè)虛析構(gòu)函數(shù);如果一個(gè)類不被設(shè)計(jì)為基類,那么這個(gè)類的析構(gòu)就應(yīng)該拒絕為虛。

建議43:絕不讓構(gòu)造函數(shù)為虛

?????????虛函數(shù)的工作機(jī)制:虛函數(shù)的多態(tài)機(jī)制是通過一張?zhí)摵瘮?shù)表來實(shí)現(xiàn)的。在構(gòu)造函數(shù)調(diào)用返回之前,虛函數(shù)表尚未建立,不能支持虛函數(shù)機(jī)制,所以構(gòu)造函數(shù)不允許設(shè)為虛。

建議44:避免在構(gòu)造/析構(gòu)函數(shù)中調(diào)用虛函數(shù)

?????????成員函數(shù)、包括虛成員函數(shù),都可以在構(gòu)造、析構(gòu)的過程中被調(diào)用。當(dāng)一個(gè)虛函數(shù)被構(gòu)造函數(shù)(包括成員變量的初始化函數(shù))或者析構(gòu)函數(shù)直接或間接地調(diào)用時(shí),調(diào)用對(duì)象就是正在構(gòu)造或者析構(gòu)的那個(gè)對(duì)象。其調(diào)用的函數(shù)是定義于自身類或者其基類的函數(shù),而不是其派生類或者最低派生類的其他基類的重寫函數(shù)。

如果在構(gòu)造函數(shù)或析構(gòu)函數(shù)中調(diào)用了一個(gè)類的虛函數(shù),那它們就變成普通函數(shù)了,失去了多態(tài)的能力。

建議45:默認(rèn)參數(shù)在構(gòu)造函數(shù)中給你帶來的喜與悲

?????????合理地使用默認(rèn)參數(shù)可以有效地減少構(gòu)造函數(shù)中的代碼冗余,讓代碼簡潔而有力。但是如果不夠小心和謹(jǐn)慎,它也會(huì)帶來構(gòu)造函數(shù)的歧義,增加你的調(diào)試時(shí)間。

建議46:區(qū)分Overloading、Overriding、Hiding之間的差異

?????????1、重載(Overloading):是指同一作用域的不同函數(shù)使用相同的函數(shù)名,但是函數(shù)的參數(shù)個(gè)數(shù)或類型不同;2、重寫(Overriding):是指在派生類中對(duì)基類中的虛函數(shù)重新實(shí)現(xiàn),即函數(shù)名和參數(shù)都一樣,只是函數(shù)的實(shí)現(xiàn)體不一樣,派生類對(duì)基類中的操作進(jìn)行個(gè)性化定制就是重寫。重寫需要注意的問題:(1)、函數(shù)的重寫與訪問層級(jí)(public、private、protected)無關(guān);(2)、const可能會(huì)使虛成員函數(shù)的重寫失效;(3)重寫函數(shù)必須和原函數(shù)具有相同的返回類型;3、隱藏(Hiding):是指派生類中的函數(shù)屏蔽基類中具有相同名字的非虛函數(shù)。

建議47:重載operator=的標(biāo)準(zhǔn)三步走

?????????1、不要讓編譯器幫你重載賦值運(yùn)算符;2、一定要檢查自賦值;3、賦值運(yùn)算符重載需返回*this的引用,引用之于對(duì)象的優(yōu)點(diǎn)在于效率,為了能夠更加靈活地使用賦值運(yùn)算符,選擇返回引用絕對(duì)是明智之舉;4、賦值運(yùn)算符重載函數(shù)不能被繼承。如果需要給類的數(shù)據(jù)成員動(dòng)態(tài)分配空間,則必須實(shí)現(xiàn)賦值運(yùn)算符。

建議48:運(yùn)算符重載,是成員函數(shù)還是友元函數(shù)

?????????運(yùn)算符重載的四項(xiàng)基本原則:(1)、不可臆造運(yùn)算符;(2)、運(yùn)算符原有操作數(shù)的個(gè)數(shù)、優(yōu)先級(jí)和結(jié)合性不能改變;(3)、操作數(shù)中至少一個(gè)是自定義類型;(4)、保持重載運(yùn)算符的自然含義。

???????? 運(yùn)算符的重載可采用兩種形式:成員函數(shù)形式和友元函數(shù)形式。(1)、重載為成員函數(shù)時(shí),已經(jīng)隱含了一個(gè)參數(shù),它就是this指針;對(duì)于雙目運(yùn)算符,參數(shù)僅有一個(gè);(2)、當(dāng)重載友元函數(shù)時(shí),將不存在隱含的參數(shù)this指針;如果運(yùn)算符被重載為友元函數(shù),那么它就獲得一種特殊的屬性,能夠接受左參數(shù)和右參數(shù)的隱式轉(zhuǎn)換,如果是成員函數(shù)版的重載則只允許右參數(shù)的隱式轉(zhuǎn)換。一般說來,建議遵守一個(gè)不成文的規(guī)定:對(duì)雙目運(yùn)算符,最好將其重載為友元函數(shù),因?yàn)檫@樣更方便些;而對(duì)于單目運(yùn)算符,則最好重載為成員函數(shù)。

建議49:有些運(yùn)算符應(yīng)該成對(duì)實(shí)現(xiàn)

???????? 為了更好地適應(yīng)使用習(xí)慣,很多運(yùn)算符重載時(shí)最好成對(duì)實(shí)現(xiàn),比如==與!=、<與>、<=與>=、+與+=、-與-=、*與*=、/與/=。

建議50:特殊的自增自減運(yùn)算符重載

?????????后綴操作重載時(shí)返回值應(yīng)該為一個(gè)const對(duì)象。

建議51:不要重載operator&&、operator||以及operator,

?????????“&&”、“||”、“,”(逗號(hào)運(yùn)算符)都具有較為特殊的行為特性,重載會(huì)改變運(yùn)算符的這些特性,進(jìn)而影響原有的習(xí)慣,所以不要去重載這三個(gè)可以重載的運(yùn)算符。

建議52:合理地使用inline函數(shù)來提高效率

?????????內(nèi)聯(lián)函數(shù)具有與宏定義相同的代碼效率,但在其他方面卻要優(yōu)于宏定義。因?yàn)閮?nèi)聯(lián)函數(shù)還遵循函數(shù)的類型和作用域規(guī)則。內(nèi)聯(lián)函數(shù)一般情況下都應(yīng)該定義在頭文件中。內(nèi)聯(lián)函數(shù)的定義分為兩種方式:(1)、顯示方式:在函數(shù)定義之前添加inline關(guān)鍵字,內(nèi)聯(lián)函數(shù)只有和函數(shù)體聲明放在一起時(shí)inline關(guān)鍵字才具有效力;(2)、隱式方式:將函數(shù)定義于類的內(nèi)部。一個(gè)給定的函數(shù)是否得到內(nèi)聯(lián),很大程度上取決于你正在使用的編譯器。

???????? 使用內(nèi)聯(lián)函數(shù)應(yīng)該注意:(1)、內(nèi)聯(lián)函數(shù)的定義必須出現(xiàn)在內(nèi)聯(lián)函數(shù)第一次被調(diào)用之前。所以,它一般會(huì)置于頭文件中;(2)、在內(nèi)聯(lián)函數(shù)內(nèi)不允許用循環(huán)語句和開關(guān)語句,函數(shù)不能過于復(fù)雜;(3)、依據(jù)經(jīng)驗(yàn),內(nèi)聯(lián)函數(shù)只適合于只有1~5行的小函數(shù);(4)、對(duì)于內(nèi)存空間有限的機(jī)器而言,慎用內(nèi)聯(lián)。過分地使用內(nèi)聯(lián)會(huì)造成函數(shù)代碼的過度膨脹,會(huì)占用太多空間;(5)、不要對(duì)構(gòu)造/析構(gòu)函數(shù)進(jìn)行內(nèi)聯(lián);(6)、大多開發(fā)環(huán)境不支持內(nèi)聯(lián)調(diào)試,所以為了調(diào)試方便,不要將內(nèi)聯(lián)優(yōu)化放在調(diào)試階段之前。

建議53:慎用私有繼承

?????????私有繼承會(huì)使基類的所有東西(包括所有的成員變量與成員函數(shù))在派生類中變成private的,也就是說基類的全部在派生類中都只能作為實(shí)現(xiàn)細(xì)節(jié),而不能成為接口。私有繼承意味著“只有implementation 應(yīng)該被繼承,interface應(yīng)該被忽略”,代表著是“is-implemented-in-terms-of”的內(nèi)在關(guān)系。通常情況下,這種關(guān)系可以采用組合的方式來實(shí)現(xiàn),并提倡優(yōu)先使用組合的方案。但是如果存在虛函數(shù)和保護(hù)成員,就會(huì)使組合方案失效,那就應(yīng)使用私有繼承。

建議54:抵制MI的糖衣炮彈

?????????MI(多重繼承)意味著設(shè)計(jì)的高復(fù)雜性、維護(hù)的高難度性,盡量少使用MI。

建議55:堤防對(duì)象切片

?????????多態(tài)的實(shí)現(xiàn)必須依靠指向同一類族的指針或引用。否則,就可能出現(xiàn)著名的對(duì)象切片(Object Slicing)問題。所以,在既有繼承又有虛函數(shù)的情況下,一定要提防對(duì)象切片問題。

建議56:在正確的場合使用恰當(dāng)?shù)奶匦?/strong>

?????????(1)、虛函數(shù):虛函數(shù)機(jī)制的實(shí)現(xiàn)是通過虛函數(shù)表和指向虛函數(shù)表的指針來完成的。關(guān)鍵字virtual告訴編譯器該函數(shù)應(yīng)該實(shí)現(xiàn)晚綁定,編譯器對(duì)每個(gè)包含虛函數(shù)的類創(chuàng)建虛函數(shù)表VTable,以放置類的虛函數(shù)地址。編譯器密碼放置了指向虛函數(shù)表的指針VPtr,當(dāng)多態(tài)調(diào)用時(shí),它會(huì)使用VPtr在VTable表中查找要執(zhí)行的函數(shù)地址;(2)、多重繼承:對(duì)于多重繼承來說,對(duì)象內(nèi)部會(huì)有多個(gè)VPrt,所以這就使偏移量計(jì)算變得復(fù)雜了,而且會(huì)使對(duì)象占用的空間和運(yùn)行時(shí)開銷都變大;(3)、虛基類:它與多重繼承的情況類似,因?yàn)樘摶惥褪菫榱硕嘀乩^承而產(chǎn)生的;(4)、運(yùn)行時(shí)類型檢測(RTTI):是我們?cè)诔绦蜻\(yùn)行時(shí)得到對(duì)象和類有關(guān)信息的保證。

建議57:將數(shù)據(jù)成員聲明為private

?????????將數(shù)據(jù)成員聲明為private是具有相當(dāng)充分的理由的:(1)、實(shí)現(xiàn)數(shù)據(jù)成員的訪問控制;(2)、在將來時(shí)態(tài)下設(shè)計(jì)程序,為之后的各種實(shí)現(xiàn)提供彈性;(3)、保持語法的一致性。

建議58:明晰對(duì)象的構(gòu)造與析構(gòu)的順序

?????????(1)、對(duì)象的構(gòu)造都是從類的最根處開始的,由深及淺,先基類后子類,層層構(gòu)造,這個(gè)順序不能改變。如果含有多個(gè)基類,那么就按照聲明順序由前及后執(zhí)行。析構(gòu)函數(shù)則嚴(yán)格按照構(gòu)造的逆序執(zhí)行;(2)、成員對(duì)象構(gòu)造函數(shù)的調(diào)用順序與成員對(duì)象的聲明順序嚴(yán)格一致,析構(gòu)順序是構(gòu)造順序的嚴(yán)格逆序。這是因?yàn)轭惖穆暶魇墙^對(duì)唯一的,而類的構(gòu)造函數(shù)可以有多個(gè),所以按照聲明才會(huì)使析構(gòu)函數(shù)得到唯一的逆序;(3)、如果繼承遇到成員對(duì)象,基類構(gòu)造函數(shù)依然會(huì)被首先調(diào)用,然后調(diào)用成員對(duì)象的構(gòu)造函數(shù)。

建議59:明了如何在主調(diào)函數(shù)啟動(dòng)前調(diào)用函數(shù)

?????????如果想在主程序main啟動(dòng)之前調(diào)用某些函數(shù),調(diào)用全局對(duì)象的構(gòu)造函數(shù)絕對(duì)是一個(gè)很不錯(cuò)的方法。因?yàn)閺母拍钌险f,全局對(duì)象是在程序開始前已經(jīng)完成了構(gòu)造,而在程序執(zhí)行之后才會(huì)實(shí)施析構(gòu)。

?

第5章???用好模板,向著GP(泛型編程)開進(jìn)

建議60:審慎地在動(dòng)、靜多態(tài)之間選擇

???????? 虛函數(shù)機(jī)制配合繼承機(jī)制,生效于運(yùn)行期,屬于晚綁定,是動(dòng)多態(tài);而模板將不同的行為和單個(gè)泛化記號(hào)相關(guān)聯(lián)發(fā)生在編譯期,屬于早綁定,被稱為靜多態(tài)。(1)、動(dòng)多態(tài):它的技術(shù)基礎(chǔ)是繼承機(jī)制和虛函數(shù),它在繼承體系之間通過虛函數(shù)表來表達(dá)共同的接口;(2)、靜多態(tài):它的技術(shù)基礎(chǔ)是模板。與動(dòng)多態(tài)相比,靜多態(tài)始終在和參數(shù)“較勁兒”,它適用于所有的類,與虛函數(shù)無關(guān)。從應(yīng)用形式上看,靜多態(tài)是發(fā)散式的,讓相同的實(shí)現(xiàn)代碼應(yīng)用于不同的場合;動(dòng)多態(tài)是收斂式的,讓不同的實(shí)現(xiàn)代碼應(yīng)用于相同的場合。從思維方式上看,前者是泛型式編程風(fēng)格,它看重的是算法的普適性;后者是對(duì)象式編程風(fēng)格,它看重的是接口與實(shí)現(xiàn)的分離度。兩者區(qū)別:(1)、動(dòng)多態(tài)的函數(shù)需要通過指針或引用傳參,而靜多態(tài)則可以傳值、傳指針、傳引用等,“適應(yīng)性”更強(qiáng);(2)、在性能上,靜多態(tài)優(yōu)于動(dòng)多態(tài),因?yàn)殪o多態(tài)無間接訪問的迂回代碼,它是單刀直入的;(3)、因?yàn)閷?shí)現(xiàn)多態(tài)的先后順序不同,所以如果出現(xiàn)錯(cuò)誤,它們拋出錯(cuò)誤的時(shí)刻也不一樣,動(dòng)多態(tài)會(huì)在運(yùn)行時(shí)報(bào)錯(cuò),而靜多態(tài)則在編譯時(shí)報(bào)錯(cuò)。

建議61:將模板的聲明和定義放置在同一個(gè)頭文件里

?????????模板類型不是一種實(shí)類型,它必須等到類型綁定后才能確定最終類型,所以在實(shí)例化一個(gè)模板時(shí),必須要能夠讓編譯器“看到”在哪里使用了模板,而且必須要看到模板確切的定義,而不僅僅是它的聲明,否則將不能正常而順利地產(chǎn)生編譯代碼。函數(shù)模板、類模板不同于一般的函數(shù)、類,它們不能像一般的方式那樣進(jìn)行聲明與定義,標(biāo)準(zhǔn)要求模板的實(shí)例化與定義體必須放在同一翻譯單元中。實(shí)現(xiàn)這一目標(biāo)有三種方法(將模板的聲明和定義都放置在同一個(gè).h文件中;按照舊有的習(xí)慣性做法來處理,聲明是聲明,實(shí)現(xiàn)是實(shí)現(xiàn),二者相互分離,但是需要包含頭文件的地方做一些改變,如,在使用模板時(shí),必須用#include “Temp.cpp”替換掉#include “Temp.h”;使用關(guān)鍵字export來定義具體的模板類對(duì)象和模板函數(shù)),但是最優(yōu)策略還是:將模板的聲明和定義都放置在同一個(gè).h文件中,雖然在某種程度上這破壞了代碼的優(yōu)雅性。

建議62:用模板替代參數(shù)化的宏函數(shù)

?????????參數(shù)化的宏函數(shù)有著兩個(gè)致命缺點(diǎn):(1)、缺乏類型檢查;(2)、有可能在不該進(jìn)行宏替換的時(shí)候進(jìn)行了替換,違背了作者的意圖。模板是實(shí)現(xiàn)代碼復(fù)用的一種工具,它可以實(shí)現(xiàn)類型參數(shù)化,達(dá)到讓代碼真正復(fù)用的目的。

建議63:區(qū)分函數(shù)模板與模板函數(shù)、類模板與模板類

?????????函數(shù)模板的重點(diǎn)在于“模板”兩個(gè)字,前面的“函數(shù)”只是一個(gè)修飾詞。其表示的是一個(gè)專門用來生產(chǎn)函數(shù)的模板。而模板函數(shù)重點(diǎn)在“函數(shù)”,表示的是用模板所生成的函數(shù)。函數(shù)模板的一般定義形式為:

???????? Template<class數(shù)據(jù)類型參數(shù)標(biāo)識(shí)符>

???????? 返回類型標(biāo)識(shí)符 函數(shù)名(數(shù)據(jù)類型參數(shù)標(biāo)識(shí)符 形參)

???????? {? //… …}

???????? 將函數(shù)模板的模板參數(shù)實(shí)例化后會(huì)生成具體的函數(shù),此函數(shù)就是模板函數(shù)。由函數(shù)模板所生成的模板函數(shù)的一般形式為:

???????? 函數(shù)名<數(shù)據(jù)類型參數(shù)標(biāo)識(shí)符>(數(shù)據(jù)類型參數(shù)標(biāo)識(shí)符 形參)

???????? 類模板是為類定義的一種模式,它使類中的一些數(shù)據(jù)成員和成員函數(shù)的參數(shù)或返回值可以取任意的數(shù)據(jù)類型。在類定義中,凡是采用標(biāo)準(zhǔn)數(shù)據(jù)類型的數(shù)據(jù)成員、成員函數(shù)的參數(shù)前面都要加上類型標(biāo)識(shí)符,在返回類型前也要進(jìn)行同樣的處理。如果類中的成員函數(shù)要在類的聲明之外定義,則它必須是模板函數(shù)。將類模板的模板參數(shù)實(shí)例化后生成的具體類,就是模板類。函數(shù)模板和類模板處于實(shí)例化之前,而模板函數(shù)或模板類則在實(shí)例化之后。

建議64:區(qū)分繼承與模板

?????????模板的長處在于處理不同類型間“千篇一律”的操作。相較于類繼承,這些類不必具有什么相同的性質(zhì)。

?

第6章???讓神秘的異常處理不再神秘

建議65:使用exception來處理錯(cuò)誤

?????????異常能:(1)、增強(qiáng)程序的健壯性;(2)、使代碼變得更簡潔優(yōu)美、更易維護(hù);(3)、錯(cuò)誤信息更靈活、豐富。

建議66:傳值throw異常,傳引用catch異常

?????????throw byvalue, catch by reference

建議67:用“throw;”來重新拋出異常

?????????對(duì)于異常的重新拋出,需要注意:(1)、重新拋出的異常對(duì)象只能出現(xiàn)在catch塊或catch調(diào)用的函數(shù)中;(2)、如果在處理代碼不執(zhí)行時(shí)碰到“throw ;”語句,將會(huì)調(diào)用terminate函數(shù)。

建議68:了解異常捕獲與函數(shù)參數(shù)傳遞之間的差異

?????????異常與函數(shù)參數(shù)的傳遞之間差異:(1)、控制權(quán);(2)、對(duì)象拷貝的次數(shù);(3)、異常類型轉(zhuǎn)換;(4)、異常類型匹配。

建議69:熟悉異常處理的代價(jià)

?????????異常處理在帶來便利的同時(shí),也會(huì)帶來時(shí)間和空間上的開銷,使程序效率降低,體積增大,同時(shí)會(huì)加大代碼調(diào)試和管理的成本。

建議70:盡量保證異常安全

?????????如果采用了異常機(jī)制,請(qǐng)盡量保證異常安全:努力實(shí)現(xiàn)強(qiáng)保證,至少實(shí)現(xiàn)基本保證。

?

第7章???用好STL這個(gè)大輪子

建議71:盡量熟悉C++標(biāo)準(zhǔn)庫

?????????C++標(biāo)準(zhǔn)庫主要包含的組件:(1)、C標(biāo)準(zhǔn)函數(shù)庫;(2)、輸入/輸出(input/output);(3)、字符串(string);(4)、容器(containers);(5)、算法(algorithms);(6)、迭代器(iterators);(7)、國際化(internationalization);(8)、數(shù)值(numerics);(9)、語言支持(languagesupport);(10)、診斷(diagnostics);(11)、通用工具(general utilities)。字符串、容器、算法、迭代器四部分采用了模板技術(shù),一般被統(tǒng)稱為STL(Standard Template Library,即標(biāo)準(zhǔn)模板庫)。

???????? 在C++標(biāo)準(zhǔn)中,STL被組織成了13個(gè)頭文件:<algorithm>、<deque>、<functional>、<iterator>、<vector>、<list>、<map>、<memory>、<numeric>、<queue>、<set>、<stack>、<utility>。

建議72:熟悉STL中的有關(guān)術(shù)語

?????????(1)、容器:是一個(gè)對(duì)象,它將對(duì)象作為元素來存儲(chǔ);(2)、泛型(Genericity):泛型就是通用,或者說是類型獨(dú)立;(3)算法:就是對(duì)一個(gè)對(duì)象序列所采取的某些操作,例如std::sort()、std::copy()、std::remove();(4)、適配器(Adaptor):是一個(gè)非常特殊的對(duì)象,它的作用就是使函數(shù)轉(zhuǎn)化為函數(shù)對(duì)象,或者是將多參數(shù)的函數(shù)對(duì)象轉(zhuǎn)化為少參數(shù)的函數(shù)對(duì)象;(5)、O(h):它是一個(gè)表示算法性能的特殊符號(hào),在STL規(guī)范中用于表示標(biāo)準(zhǔn)庫算法和容器操作的最低性能極限;(6)、迭代器:是一種可以當(dāng)做通用指針來使用的對(duì)象,迭代器可以用于元素遍歷、元素添加和元素刪除。

建議73:刪除指針的容器時(shí)避免資源泄露

?????????STL容器雖然智能,但尚不能擔(dān)當(dāng)刪除它們所包含指針的這一責(zé)任。所以,在要?jiǎng)h除指針的容器時(shí)須避免資源泄露:或者在容器銷毀前手動(dòng)刪除容器中的每個(gè)指針,或者使用智能引用計(jì)數(shù)指針對(duì)象(比如Boost的shared_ptr)來代替普通指針。

建議74:選擇合適的STL容器

?????????容器分為:(1)、標(biāo)準(zhǔn)STL序列容器:vector、string、deque和list;(2)、標(biāo)準(zhǔn)STL關(guān)聯(lián)容器:set、multiset、map和multimap;(3)、非標(biāo)準(zhǔn)序列容器:slist(單向鏈表)和rope(重型字符串);(4)、非標(biāo)準(zhǔn)關(guān)聯(lián)容器:hash_set、hash_multiset、hash_map和hash_multimap;(5)、標(biāo)準(zhǔn)非STL容器:數(shù)組、bitset、valarray、stack、queue和priority_queue。

建議75:不要在STL容器中存儲(chǔ)auto_ptr對(duì)象

?????????auto_ptr是C++標(biāo)準(zhǔn)中提供的智能指針,它是一個(gè)RAII對(duì)象,它在初始化時(shí)獲得資源,析構(gòu)時(shí)自動(dòng)釋放資源。C++標(biāo)準(zhǔn)中規(guī)定:STL容器元素必須能夠進(jìn)行拷貝構(gòu)造和賦值操作。禁止在STL容器中存儲(chǔ)auto_ptr對(duì)象原因有兩個(gè):(1)、auto_ptr拷貝操作不安全,會(huì)使原指針對(duì)象變NULL;(2)、嚴(yán)重影響代碼的可移植性。

建議76:熟悉刪除STL容器中元素的慣用法

?????????(1)、刪除容器中具有特定值的元素:如果容器是vector、string或deque,使用erase-remove的慣用法(remove只會(huì)將不應(yīng)該刪除的元素前移,然后返回一個(gè)迭代器,該迭代器指向的是那個(gè)應(yīng)該刪除的元素,所以如果要真正刪除這一元素,在調(diào)用remove之后還必須調(diào)用erase);如果容器時(shí)list,使用list::remove;如果容器是標(biāo)準(zhǔn)關(guān)聯(lián)容器,使用它的erase成員函數(shù);(2)、刪除容器中滿足某些條件的所有元素:如果容器是vector、string或deque,使用erase-remove_if慣用法;如果容器是list,使用list::remove_if;如果容器是標(biāo)準(zhǔn)關(guān)聯(lián)容器,使用remove_copy_if & swap組合算法,或者自己寫一個(gè)遍歷刪除算法。

建議77:小心迭代器的失效

???????? 迭代器是一個(gè)對(duì)象,其內(nèi)存大小為12(sizeof(vector<int>::iterator),vs2010,32bit)。引起迭代器失效的最主要操作就是插入、刪除。對(duì)于序列容器(如vector和deque),插入和刪除操作可能會(huì)使容器的部分或全部迭代器失效。因?yàn)関ector和deque必須使用連續(xù)分配的內(nèi)存來存儲(chǔ)元素,向容器中添加一個(gè)元素可能會(huì)導(dǎo)致后面鄰接的內(nèi)存沒有可用的空閑空間而引起存儲(chǔ)空間的重新分配。一旦這種情況發(fā)生,容器中的所有的迭代器就會(huì)全部失效。

建議78:盡量使用vector和string代替動(dòng)態(tài)分配數(shù)組

?????????相較于內(nèi)建數(shù)組,vector和string具有幾方面的優(yōu)點(diǎn):(1)、它們能夠自動(dòng)管理內(nèi)存;(2)、它們提供了豐富的接口;(3)、與C的內(nèi)存模型兼容;(4)、集眾人智慧之大成。

建議79:掌握vector和string與C語言API的通信方式

?????????使用vector::operator[]和string::c_str是實(shí)現(xiàn)STL容器與C語言API通信的最佳方式。

建議80:多用算法調(diào)用,少用手寫循環(huán)

?????????用算法調(diào)用代替手工編寫的循環(huán),具有幾方面的優(yōu)點(diǎn):(1)、效率更高;(2)、不易出錯(cuò);(3)、可維護(hù)性更好。

?

第二部分 編碼習(xí)慣和規(guī)范篇

?

第8章???讓程序正確執(zhí)行

建議81:避免無意中的內(nèi)部數(shù)據(jù)裸露

???????? 對(duì)于const成員函數(shù),不要返回內(nèi)部數(shù)據(jù)的句柄,因?yàn)樗鼤?huì)破壞封裝性,違反抽象性,造成內(nèi)部數(shù)據(jù)無意中的裸露,這會(huì)出現(xiàn)很多“不可思議”的情形,比如const對(duì)象的非常量性。

建議82:積極使用const為函數(shù)保駕護(hù)航

?????????const的真正威力體現(xiàn)在幾個(gè)方面:(1)、修飾函數(shù)形式的參數(shù):const只能修飾輸入?yún)?shù),對(duì)于內(nèi)置數(shù)據(jù)類型的輸入?yún)?shù),不要將“值傳遞”的方式改為“const 引用傳遞”;(2)、修飾函數(shù)返回值;(3)、修飾成員函數(shù):用const修飾成員函數(shù)的目的是提高程序的健壯性。const成員函數(shù)不允許對(duì)數(shù)據(jù)成員進(jìn)行任何修改。

???????? 關(guān)于const成員函數(shù),須遵循幾個(gè)規(guī)則:(1)、const對(duì)象只能訪問const成員函數(shù),而非const對(duì)象可以訪問任意的成員函數(shù);(2)、const對(duì)象的成員是不可修改的,然而const對(duì)象通過指針維護(hù)的對(duì)象卻是可以修改的;(3)、const成員函數(shù)不可以修改對(duì)象的數(shù)據(jù),不管對(duì)象是否具有const性質(zhì)。

建議83:不要返回局部變量的引用

?????????局部變量的引用是一件不太靠譜的事兒,所以盡量避免讓函數(shù)返回局部變量的引用。同時(shí)也不要返回new生成對(duì)象的引用,因?yàn)檫@樣會(huì)讓代碼層次混亂,讓使用者苦不堪言。

建議84:切忌過度使用傳引用代替?zhèn)鲗?duì)象

?????????相較于傳對(duì)象,傳引用的優(yōu)點(diǎn):它減少了臨時(shí)對(duì)象的構(gòu)造與析構(gòu),所以更具效率。但須審慎地使用傳引用替代傳對(duì)象,必須傳回內(nèi)部對(duì)象時(shí),就傳對(duì)象,勿傳引用。

建議85:了解指針參數(shù)傳遞內(nèi)存中的玄機(jī)

?????????用指針參數(shù)傳回一塊動(dòng)態(tài)申請(qǐng)的內(nèi)存,是很常見的一種需求。然而如果不甚小心,就很容易造成嚴(yán)重錯(cuò)誤:程序崩潰+內(nèi)存泄露,解決之道就是用指針的指針來傳遞,或者換種內(nèi)存?zhèn)鬟f方式,用返回值來傳遞。

建議86:不要講函數(shù)參數(shù)作為工作變量

?????????工作變量,就是在函數(shù)實(shí)現(xiàn)中使用的變量。應(yīng)該防止將函數(shù)參數(shù)作為工作變量,而對(duì)于那些必須改變的參數(shù),最好先用局部變量代替之,最后再將該局部變量的內(nèi)容賦給該參數(shù),這樣在一定程度上保護(hù)了數(shù)據(jù)的安全。

建議87:躲過0值比較的層層陷阱

?????????(1)、0在不在該類型數(shù)據(jù)的取值范圍內(nèi)?(2)、浮點(diǎn)數(shù)不存在絕對(duì)0值,所以浮點(diǎn)零值比較需特殊處理;(3)區(qū)分比較操作符==與賦值操作符=,切忌混淆。

建議88:不要用reinterpret_cast去迷惑編譯器

?????????reinterpret_cast,簡單地說就是保持二進(jìn)制位不變,用另一種格式來重新解釋,它就是C/C++中最為暴力的類型轉(zhuǎn)換,所實(shí)現(xiàn)的是一個(gè)類型到一個(gè)毫不相關(guān)、完全不同類型的映射。reiterpret_cast僅僅重新解釋了給出對(duì)象的比特模型,它是所有類型轉(zhuǎn)換中最危險(xiǎn)的。盡量避免使用reinterpret_cast,除非是在其他轉(zhuǎn)換都無效的非常情形下。

建議89:避免對(duì)動(dòng)態(tài)對(duì)象指針使用static_cast

?????????在類層次結(jié)構(gòu)中,用static_cast完成基類和子類指針(或引用)的下行轉(zhuǎn)換是不安全的。所以盡量避免對(duì)動(dòng)態(tài)對(duì)象指針使用static_cast,可以用dynamic_cast來代替,或者優(yōu)化設(shè)計(jì),重構(gòu)代碼。

建議90:盡量少應(yīng)用多態(tài)性數(shù)組

?????????多態(tài)性數(shù)組一方面會(huì)涉及C++時(shí)代的基類指針與派生類指針之間的替代問題,同時(shí)也會(huì)涉及C時(shí)代的指針運(yùn)算,而且常會(huì)因?yàn)檫@二者之間的不協(xié)調(diào)引發(fā)隱蔽的Bug。

建議91:不要強(qiáng)制去除變量的const屬性

?????????在C++中,const_cast<T*>(a)一般用于從一個(gè)類中去除以下這些屬性:const、volatile和_unaligned.強(qiáng)制去除變量的const屬性雖然可以帶來暫時(shí)的便利,但這不僅增加了錯(cuò)誤修改變量的幾率,而且還可能會(huì)引發(fā)內(nèi)存故障。

?

第9章???提高代碼的可讀性

建議92:盡量使代碼版面整潔優(yōu)雅

?????????(1)、避免代碼過長;(2)、代碼縮進(jìn)和對(duì)齊;(3)、空行分隔段落;(4)、使用空格;(5)、語句行。

建議93:給函數(shù)和變量起一個(gè)“能說話”的名字

?????????(1)、名稱必須直觀,可望文生義,不必解碼;(2)、長度要符合“min_length && max_information”(最小名長度最大信息量)的原則;(3)、與整體風(fēng)格保持一致;(4)、變量名稱應(yīng)該是一個(gè)“名詞”,或者是“形容詞+名詞”;而函數(shù)名稱應(yīng)該是“動(dòng)詞+名詞”的組合;(5)、杜絕僅靠大小寫來區(qū)分的名稱標(biāo)示符;(6)、變量名之前附加前綴用來識(shí)別變量類型;(7)、C++類或結(jié)構(gòu)的成員變量附加前綴“m_”;全局變量名稱附加前綴“g_”;(8)、單字符變量只能用作循環(huán)變量;(9)、類名采用“C+首字母大寫的單詞”形式來命名。

建議94:合理地添加注釋

?????????(1)、使用統(tǒng)一的注釋方法為每個(gè)層次的代碼塊添加注釋;(2)、避免不必要的注釋;(3)、掌握代碼注釋量的一個(gè)度;(4)、邊寫代碼加邊注釋;(5)、注釋要簡明而準(zhǔn)確;(6)、注意特有標(biāo)簽的特有作用。

建議95:為源代碼設(shè)置一定的目錄結(jié)構(gòu)

?????????如果一個(gè)軟件所涉及的文件數(shù)目比較多,通常要將其進(jìn)行劃分,為其設(shè)置一定的目錄結(jié)構(gòu),以便于維護(hù),如include、lib、src、doc、release、debug。

建議96:用有意義的標(biāo)識(shí)代替Magic Numbers

?????????用宏或常量替代信息含量較低的MagicNumbers,絕對(duì)是一個(gè)好習(xí)慣,這樣可提高代碼的可讀性與可維護(hù)性。

建議97:避免使用“聰明的技巧”

建議98:運(yùn)算符重載時(shí)堅(jiān)持其通用的含義

建議99:避免嵌套過深與函數(shù)過長

建議100:養(yǎng)成好習(xí)慣,從現(xiàn)在做起

?

第10章???讓代碼運(yùn)行得再快些

建議101:用移位實(shí)現(xiàn)乘除法運(yùn)算

?????????在大部分的C/C++編譯器中,用移位的方法比直接調(diào)用乘除法子程序生成代碼的效率要高。只要是乘以或除以一個(gè)整數(shù)常量,均可用移位的方法得到結(jié)果,如a=a*9可以拆分成a=a*(8+1),即a=a(a<<3)+a。移位只對(duì)整數(shù)運(yùn)算起作用。

建議102:優(yōu)化循環(huán),提高效率

?????????應(yīng)當(dāng)將最長的循環(huán)放在最內(nèi)層,最短的循環(huán)放在最外層,以減少CPU跨切循環(huán)層的次數(shù),提高效率。

建議103:改造switch語句

?????????對(duì)于case的值,推薦按照它們發(fā)生的相對(duì)頻率來排序,把最可能發(fā)生的情況放在第一位,最不可能的情況放在最后。

建議104:精簡函數(shù)參數(shù)

?????????函數(shù)在調(diào)用時(shí)會(huì)建立堆棧來存儲(chǔ)所需的參數(shù)值,因此函數(shù)的調(diào)用負(fù)擔(dān)會(huì)隨著參數(shù)列表的增長而增加。所以,參數(shù)的個(gè)數(shù)會(huì)影響進(jìn)棧出棧的次數(shù),當(dāng)參數(shù)很多的時(shí)候,這樣的操作就會(huì)花費(fèi)很長的時(shí)間。因此,精簡函數(shù)參數(shù),減少參數(shù)個(gè)數(shù)可以提高函數(shù)調(diào)用的效率。如果精簡后的參數(shù)還是比較多,那么可以把參數(shù)列表封裝進(jìn)一個(gè)單獨(dú)的類中,并且可以通過引用進(jìn)行傳遞。

建議105:謹(jǐn)慎使用內(nèi)嵌匯編

?????????匯編語言與其他高級(jí)語言相比,更接近機(jī)器語言,效率更高,所以在應(yīng)用程序中如果碰到那些對(duì)時(shí)間要求苛刻的部分,可以采用匯編語言來重寫。

建議106:努力減少內(nèi)存碎片

?????????經(jīng)常性地動(dòng)態(tài)分配和釋放內(nèi)存會(huì)造成堆碎片,尤其是應(yīng)用程序分配的是很小的內(nèi)存塊時(shí)。避免堆碎片:(1)、盡可能少地使用動(dòng)態(tài)內(nèi)存,在大多數(shù)情況下,可以使用靜態(tài)或自動(dòng)儲(chǔ)存,或者使用STL容器,減少對(duì)動(dòng)態(tài)內(nèi)存的依賴;(2)、盡量分配和重新分配大塊的內(nèi)存塊,降低內(nèi)存碎片發(fā)生的幾率。內(nèi)存碎片會(huì)影響程序執(zhí)行的效率。

建議107:正確地使用內(nèi)聯(lián)函數(shù)

?????????內(nèi)聯(lián)(inline)函數(shù)既能夠去除函數(shù)調(diào)用所帶來的效率負(fù)擔(dān),又能夠保留一般函數(shù)的優(yōu)點(diǎn)。只有當(dāng)函數(shù)非常短小的時(shí)候使用inline才能得到預(yù)想中的效果。對(duì)于編譯器來說,函數(shù)內(nèi)聯(lián)與否取決于以下關(guān)鍵性的因素:(1)、函數(shù)體的大小;(2)、是否有局部對(duì)象被聲明;(3)、函數(shù)的復(fù)雜性(是否存在函數(shù)調(diào)用、遞歸等)。

建議108:用初始化取代賦值

?????????以用戶初始化代替賦值,可以使效率得到較大的提升,因?yàn)檫@樣可以避免一次賦值函數(shù)operator =的調(diào)用。因此,當(dāng)我們?cè)谫x值和初始化之間進(jìn)行選擇時(shí),初始化應(yīng)該是首選。需要注意的是,對(duì)基本的內(nèi)置數(shù)據(jù)類型而言,初始化和賦值之間是沒有差異的,因?yàn)閮?nèi)置類型沒有構(gòu)造和析構(gòu)過程。

建議109:盡可能地減少臨時(shí)對(duì)象

?????????臨時(shí)對(duì)象產(chǎn)生的主要情形及避免方法:(1)、參數(shù):采用傳常量引用或指針取代傳值;(2)、前綴或后綴:優(yōu)先采用前綴操作;(3)、參數(shù)轉(zhuǎn)換:盡量避免這種轉(zhuǎn)換;(4)、返回值:遵循single-entry/single-exit原則,避免同一個(gè)函數(shù)中存在多個(gè)return語句。

建議110:最后再去優(yōu)化代碼

?????????在進(jìn)行代碼優(yōu)化之前,需要知道:(1)、算法是否正確;(2)、如何在代碼優(yōu)化和可讀性之間進(jìn)行選擇;(3)、該如何優(yōu)化:代碼分析(profiling)工具;(4)、如何選擇優(yōu)化方向:先算法,再數(shù)據(jù)結(jié)構(gòu),最后才是實(shí)現(xiàn)細(xì)節(jié)。

?

第11章???零碎但重要的其他建議

建議111:采用相對(duì)路徑包含頭文件

?????????一個(gè)“點(diǎn)”(“.\”)代表的是當(dāng)前目錄所在的路徑,兩個(gè)“點(diǎn)”(“..\”)代表的是相對(duì)于當(dāng)前目錄的上一次目錄路徑。

???????? 當(dāng)寫#include語句時(shí),推薦使用相對(duì)路徑;此外,要注意使用比較通用的正斜線“/”,而不要使用僅在Windows下可用的反斜線“\”。

建議112:讓條件編譯為開發(fā)出力

?????????條件編譯中的預(yù)處理命令主要包括:#if、#ifndef、#ifdef、#endif和#undef等,它們的主要功能是在程序編譯時(shí)進(jìn)行有選擇性的挑選,注釋掉一些指定的代碼,以達(dá)到版本控制、防止對(duì)文件重復(fù)包含等目的。

建議113:使用.inl文件讓代碼整潔可讀

?????????.inl文件是內(nèi)聯(lián)函數(shù)的源文件,.inl文件還可用于模板的定義。.inl文件可以將頭文件與內(nèi)聯(lián)函數(shù)的復(fù)雜定義隔離開來,使代碼整潔可讀,如果將其用于模板定義,這一優(yōu)點(diǎn)更加明顯。

建議114:使用斷言來發(fā)現(xiàn)軟件問題

?????????斷言assert,只會(huì)在調(diào)試模式下生成代碼,而在Release版本中則直接無視之。

assert(pData!= NULL && “In Function DataProcess”)assert(0&& “MSG_HANDLE_ERROR”) 建議115:優(yōu)先選擇編譯和鏈接錯(cuò)誤

?

?????????靜態(tài)檢查:編譯器必須檢查源程序是否符合源語言規(guī)定的語法和語義要求,靜態(tài)檢查的主要工作就是語義分析,它是獨(dú)立于數(shù)據(jù)和控制流的,可信度相對(duì)較高,而且不會(huì)增加程序的運(yùn)行時(shí)開銷。

???????? 動(dòng)態(tài)檢查:是在運(yùn)行時(shí)刻對(duì)程序的正確性、安全性等做檢查,比如內(nèi)存不足、溢出、數(shù)組越界、除0等,這類檢查對(duì)于數(shù)據(jù)和控制流比較依賴。

???????? C/C++語言屬于一種靜態(tài)語言。一個(gè)設(shè)計(jì)較好的C++程序應(yīng)該是較少地依賴動(dòng)態(tài)檢查,更多地依賴靜態(tài)檢查。

建議116:不放過任何一條編譯器警告

?????????強(qiáng)烈建議:(1)、把編譯器的警告級(jí)別調(diào)至最高;(2)、不要放過編譯器的任何一條警告信息。

建議117:盡量減少文件之間的編譯依賴

?????????不要在頭文件中直接包含要使用的類的頭文件(除了標(biāo)準(zhǔn)庫),直接包含頭文件這種方式相對(duì)簡單方便,但是會(huì)耗費(fèi)大量的編譯時(shí)間。推薦使用類的前向聲明來減少文件直接的編譯依賴。用對(duì)類聲明的依賴替代對(duì)類定義的依賴,這是減少編譯依賴的原則。

???????? 為了加快編譯進(jìn)程,減少時(shí)間的浪費(fèi),我們應(yīng)該盡量減少頭文件依賴,其中的可選方案包括前向聲明、柴郡貓技術(shù)等。

建議118:不用在頭文件中使用using

?????????名空間是C++提供的一種機(jī)制,可以有效地避免函數(shù)名污染。然而在應(yīng)用時(shí)要十分注意:任何情況下都不應(yīng)在頭文件中使用“using namespace XXX”這樣的語句,而應(yīng)該在定義時(shí)直接用全稱。

建議119:劃分全局名空間避免名污染

?????????使用自己的名空間將全局名空間合理劃分,會(huì)有效地減少名污染問題,因?yàn)?#xff0c;不要簡單地將所有的符號(hào)和名稱統(tǒng)統(tǒng)扔進(jìn)全局名空間里。

?

第三部分 程序架構(gòu)和思想篇

?

第12章???面向?qū)ο蟮念愒O(shè)計(jì)

建議120:堅(jiān)持“以行為為中心”的類設(shè)計(jì)

?????????“以數(shù)據(jù)為中心”關(guān)注類的內(nèi)部數(shù)據(jù)結(jié)構(gòu),習(xí)慣將private類型的數(shù)據(jù)寫在前面,而將public類型的函數(shù)寫在后面。

???????? “以行為為中心”關(guān)注的重心放在了類的服務(wù)和接口上,習(xí)慣將public類型的函數(shù)寫在前面,而將private類型的數(shù)據(jù)寫在后面。

建議121:用心做好類設(shè)計(jì)

?????????在設(shè)計(jì)一個(gè)類時(shí),首先,類的設(shè)計(jì)就是數(shù)據(jù)類型的設(shè)計(jì),在數(shù)據(jù)類型的設(shè)計(jì)中,(1)、類應(yīng)該如何創(chuàng)建和銷毀呢?這會(huì)影響到類的構(gòu)造函數(shù)和析構(gòu)函數(shù)的設(shè)計(jì)。首先應(yīng)該確定類是否需要分配資源,如果需要,還要確定這些資源又該如何釋放。(2)、類是否需要一個(gè)無參構(gòu)造函數(shù)?如果需要,而恰恰此時(shí)這個(gè)類已經(jīng)有了構(gòu)造函數(shù),那么我們就得顯示地寫一個(gè)。(3)、類需要復(fù)制構(gòu)造函數(shù)嗎?其參數(shù)上加上了const修飾嗎?它是用來定義這個(gè)類傳值(pass-by-value)的具體實(shí)現(xiàn)的。(4)、所有的數(shù)據(jù)成員是不是都已經(jīng)在構(gòu)造函數(shù)中完成了初始化呢?(5)、類需要賦值操作符嗎?賦值操作符能正確地將對(duì)象賦給對(duì)象本身嗎?它與初始化有什么不同?其參數(shù)上加上了const修飾嗎?(6)、類的析構(gòu)函數(shù)需要設(shè)置為virtual嗎?(7)、類中哪些值得組合是合法的?合法值的限定條件是什么?在成員函數(shù)內(nèi)部是否對(duì)變量值得合法性做了檢查?其次,類的設(shè)計(jì)是對(duì)現(xiàn)實(shí)對(duì)象進(jìn)行抽象的一個(gè)過程。再次,數(shù)據(jù)抽象的過程其實(shí)是綜合考慮各方面因素進(jìn)行權(quán)衡的一個(gè)過程。

建議122:以指針代替嵌入對(duì)象或引用

?????????設(shè)計(jì)類的數(shù)據(jù)成員時(shí),可以有三種選擇:(1)、嵌入對(duì)象;(2)、使用對(duì)象引用;(3)、使用對(duì)象指針。

???????? 如果在類數(shù)據(jù)成員中使用到了自定義數(shù)據(jù)類型,使用指針是一個(gè)較為明智的選擇,它有以下幾方面的優(yōu)點(diǎn):(1)、成員對(duì)象類型的變化不會(huì)引起包含類的重編譯;(2)、支持惰性計(jì)算,不創(chuàng)建不使用的對(duì)象,效率更高;(3)、支持?jǐn)?shù)據(jù)成員的多態(tài)行為。

建議123:努力將接口最小化且功能完善

?????????類接口的目標(biāo)是完整且最小。精簡接口函數(shù)個(gè)數(shù),使每一個(gè)函數(shù)都具有代表性,并且使其功能恰好覆蓋class的智能,同時(shí)又可以獲得接口精簡所帶來的好處:(1)、利于理解、使用,維護(hù)成本也相對(duì)較低;(2)、可以縮小頭文件長度,并縮短編譯時(shí)間。

建議124:讓類的數(shù)據(jù)隱藏起來

?????????堅(jiān)持?jǐn)?shù)據(jù)封裝,堅(jiān)持信息隱藏,杜絕公有、保護(hù)屬性的存在(數(shù)據(jù)成員私有、柴郡貓技術(shù))。

建議125:不要讓成員函數(shù)破壞類的封裝性

?????????小心類的成員函數(shù)返回屬性變量的“直接句柄”,它會(huì)破壞辛辛苦苦搭建維護(hù)的封裝性,一種方法,將函數(shù)的返回值加上const修飾。

建議126:理解“virtual + 訪問限定符”的深層含義

?????????virtual關(guān)鍵字是C++中用于實(shí)現(xiàn)多態(tài)的重要機(jī)制,其核心理念就是通過基類訪問派生類定義的函數(shù)。

???????? (1)、基類中的一個(gè)虛擬私有成員函數(shù),表示實(shí)現(xiàn)細(xì)節(jié)是可以被派生類修改的;(2)、基類中的一個(gè)虛擬保護(hù)成員函數(shù),表示實(shí)現(xiàn)細(xì)節(jié)是必須被派生類修改的;(3)、基類中的一個(gè)虛擬公有成員函數(shù),則表示這是一個(gè)接口,不推薦,建議用protected virtual 來替換。

建議127:謹(jǐn)慎恰當(dāng)?shù)厥褂糜言獧C(jī)制

?????????通常說來,類中的私有成員一般是不允許外面訪問的。但是友元可以超脫這條禁令,它可以訪問該類的私有成員。所帶來的最大好處就是避免了類成員函數(shù)的頻繁調(diào)用,節(jié)約了處理器的開銷,提高了程序的效率。但是,通常,大家認(rèn)為“友元破壞了類的封裝性”。采用友元機(jī)制,一般是基于這樣的需求:一個(gè)類的部分成員需要對(duì)個(gè)別其他類公開。

建議128:控制對(duì)象的創(chuàng)建方式

?????????棧和堆是對(duì)象的主要分布區(qū),它們對(duì)應(yīng)著兩種基本的對(duì)象創(chuàng)建方式:以new方式手動(dòng)管理的堆創(chuàng)建和只需聲明就可使用的棧創(chuàng)建。

???????? 控制對(duì)象的創(chuàng)建方式:(1)、要求在堆中建立對(duì)象:為了執(zhí)行這種限制,必須找到一種方法保證調(diào)用new是建立對(duì)象的唯一手段。非堆對(duì)象是在定義它時(shí)自動(dòng)構(gòu)造的,而且是在生存期結(jié)束時(shí)自動(dòng)釋放的。將析構(gòu)函數(shù)聲明為private,而構(gòu)造函數(shù)保持為public;(2)、禁止在堆中建立對(duì)象:要禁止調(diào)用new來建立對(duì)象,可以通過將operator new函數(shù)聲明為private來實(shí)現(xiàn)。

建議129:控制實(shí)例化對(duì)象的個(gè)數(shù)

?????????當(dāng)實(shí)例化對(duì)象唯一時(shí),采用設(shè)計(jì)模式中的單件模式;當(dāng)實(shí)例化對(duì)象為N(N>0)個(gè)時(shí),設(shè)置計(jì)數(shù)變量是一個(gè)思路。

建議130:區(qū)分繼承與組合

?????????(1)、繼承:C++的“繼承”特性可以提高程序的可復(fù)用性。繼承規(guī)則:若在邏輯上B是一種A,并且A的所有功能和屬性對(duì)B而言都有意義,則允許B繼承A的功能和屬性。繼承易于修改或擴(kuò)展那些被復(fù)用的實(shí)現(xiàn)。但它的這種“白盒復(fù)用”卻容易破壞封裝性,因?yàn)檫@會(huì)將父類的實(shí)現(xiàn)細(xì)節(jié)暴露給子類。當(dāng)父類實(shí)現(xiàn)更改時(shí),子類也不得不隨之更改,所以,從父類繼承來的實(shí)現(xiàn)將不能在運(yùn)行期間進(jìn)行改變;(2)、組合:在邏輯上表示的是“有一個(gè)(Hase-A)”的關(guān)系,即A是B的一部分。組合屬于“黑盒”復(fù)用,被包含對(duì)象的內(nèi)部細(xì)節(jié)對(duì)外是不可見的。所以,它的封裝性相對(duì)較好,實(shí)現(xiàn)上的相互依賴性比較小。并且可以通過獲取指向其他的具有相同類型的對(duì)象引用,在運(yùn)行期間動(dòng)態(tài)地定義組合。而其缺點(diǎn)就是致使系統(tǒng)中的對(duì)象過多。Is-A關(guān)系用繼承表達(dá),Has-A關(guān)系用組合表達(dá),優(yōu)先使用(對(duì)象)組合。

建議131:不要將對(duì)象的繼承關(guān)系擴(kuò)展至對(duì)象容器

?????????A是B的基類,B是一種A,但是B的容器卻不能是這種A的容器。

建議132:杜絕不良繼承

?????????在繼承體系中,派生類對(duì)象必須是可以取代基類對(duì)象的。

建議133:將RAII作為一種習(xí)慣

?????????RAII(ResourceAcquisition Is Initialization),資源獲取即初始化,RAII是C++語言的一種管理資源、避免泄露的慣用方法。RAII的做法是使用一個(gè)對(duì)象,在其構(gòu)造時(shí)獲取資源,在對(duì)象生命周期中控制對(duì)象資源的訪問,使之始終保持有效,最后再對(duì)象析構(gòu)時(shí)釋放資源。實(shí)現(xiàn)這種功能的類即采用了RAII方式,這樣的類被稱為封裝類。

建議134:學(xué)習(xí)使用設(shè)計(jì)模式

?????????設(shè)計(jì)模式是用來“封裝變化、降低耦合”的工具,它是面向?qū)ο笤O(shè)計(jì)時(shí)代的產(chǎn)物,其本質(zhì)就是充分運(yùn)用面向?qū)ο蟮娜齻€(gè)特性(即:封裝、繼承和多態(tài)),并進(jìn)行靈活的組合。

建議135:在接口繼承和實(shí)現(xiàn)繼承中做謹(jǐn)慎選擇

?????????在接口繼承和實(shí)現(xiàn)繼承之間進(jìn)行選擇時(shí),需要考慮的一個(gè)因素就是:基類的默認(rèn)版本。對(duì)于那些無法提供默認(rèn)版本的函數(shù)接口我們選擇函數(shù)接口繼承;而對(duì)于那些能夠提供默認(rèn)版本的,函數(shù)實(shí)現(xiàn)繼承就是最佳選擇。

建議136:遵循類設(shè)計(jì)的五項(xiàng)基本原則

?????????(1)、單一職責(zé)原則(SRP):一個(gè)類,最好只做一件事。SRP可以看作是低耦合、高內(nèi)聚在面向?qū)ο笤瓌t上的引申;(2)、開閉原則(OCP):對(duì)擴(kuò)展開放,對(duì)更改關(guān)閉,應(yīng)該能夠不用修改原有類就能擴(kuò)展一個(gè)類的行為;(3)、替換原則(LSP ):子類應(yīng)當(dāng)可以替換父類并出現(xiàn)在父類能夠出現(xiàn)的任何地方。反過來則不成立,子類可以替換基類,但是基類不一定能替換子類;(4)、依賴倒置原則(DIP):高層模塊不依賴于底層模塊,而是二者都依賴于抽象,即抽象不依賴于具體,具體依賴于抽象。依賴一定會(huì)存在類與類、模塊與模塊之間。當(dāng)兩個(gè)模塊之間存在緊密的耦合關(guān)系時(shí),最好的方法就是分離接口和實(shí)現(xiàn):在依賴之間定義一個(gè)抽象的接口使得高層模塊調(diào)用接口,底層模塊實(shí)現(xiàn)接口的定義,從而有效控制耦合關(guān)系,達(dá)到依賴于抽象的設(shè)計(jì)目的;(5)、接口分離原則(ISP):使用多個(gè)小的專門的接口,而不要使用一個(gè)大的總接口。接口有效地將細(xì)節(jié)和抽象隔離開來,體現(xiàn)了對(duì)抽象編程的一切好處,接口隔離強(qiáng)調(diào)接口的單一性。分離的手段主要有兩種方式:一個(gè)是利用委托分離接口,另一個(gè)是利用多重繼承分離接口。

?

第13章??返璞歸真的程序設(shè)計(jì)

建議137:用表驅(qū)動(dòng)取代冗長的邏輯選擇

?????????表驅(qū)動(dòng)法(Table drivenmethod),是一種不必用很多的邏輯語句(if或case)就可以把表中信息找出來的方法。它是一種設(shè)計(jì)模式,可用來代替復(fù)雜的if/else或switch-case邏輯判斷。

建議138:為應(yīng)用設(shè)定特性集

?????????對(duì)待C++高級(jí)特性的態(tài)度一定要謹(jǐn)慎,是否有必要使用多重繼承、異常、RTTI、模板及模板元編程,一定要做一個(gè)審慎的評(píng)估,以便為應(yīng)用選擇合適的特征集,避免使用過分復(fù)雜的設(shè)計(jì)和功能,否則將會(huì)使得代碼難于理解和維護(hù)。

建議139:編碼之前需三思

?????????在讓電腦運(yùn)行你的程序之前,先讓你的大腦編譯運(yùn)行。

建議140:重構(gòu)代碼

?????????重構(gòu)無止境,重構(gòu)你的代碼,精雕細(xì)琢,千錘百煉。

建議141:透過表面的語法挖掘背后的語義

建議142:在未來時(shí)態(tài)下開發(fā)C++程序

?????????在未來時(shí)態(tài)下開發(fā)C++程序,需要考慮代碼的可重用性、可維護(hù)性、健壯性,以及可移植性。

建議143:根據(jù)你的目的決定造不造輪子

?????????在編程語言中這些輪子表現(xiàn)為大量的通用類和庫。在工程實(shí)踐中,不要重復(fù)造輪子;而在學(xué)習(xí)研究中,鼓勵(lì)重復(fù)造輪子。

建議144:謹(jǐn)慎在OO與GP之間選擇

?????????面向?qū)ο?OO)和泛型編程(GP)是C++提供給程序員的兩種矛盾的思考模式。OO是我嗎難以割舍的設(shè)計(jì)原則,世界是對(duì)象的,我們面向?qū)ο蠓治觥⒃O(shè)計(jì)、編程;而泛型編程則關(guān)注于產(chǎn)生通用的軟件組件,讓這些組件在不同的應(yīng)用場合都能很容易的重用。

建議145:讓內(nèi)存管理理念與時(shí)俱進(jìn)

?????????學(xué)習(xí)STL allocator,更新內(nèi)存管理理念。

建議146:從大師的代碼中學(xué)習(xí)編程思想與技藝

?????????閱讀代碼需要方法:剛開始不要糾結(jié)于代碼的細(xì)節(jié),將關(guān)注的重點(diǎn)放在代碼的高層結(jié)構(gòu)上,理解代碼的構(gòu)建過程;之后,再有重點(diǎn)的深入研究,理解對(duì)象的構(gòu)造,明晰算法的步驟,嘗試著深入理解其中的繁雜細(xì)節(jié)。

建議147:遵循自然而然的C++風(fēng)格

建議148:了解C++語言的設(shè)計(jì)目標(biāo)與原則

建議149:明確選擇C++的理由

總結(jié)

以上是生活随笔為你收集整理的编写高质量代码改善C++程序的150个建议的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。

如果覺得生活随笔網(wǎng)站內(nèi)容還不錯(cuò),歡迎將生活随笔推薦給好友。

狠狠干激情 | 日日日天天天 | 日韩91在线 | 国产精品视频永久免费播放 | 人人爱爱 | 高清av免费一区中文字幕 | 在线导航av| 在线精品视频免费播放 | 五月婷婷在线视频观看 | 成人精品视频 | 亚洲精品黄色片 | 蜜臀aⅴ国产精品久久久国产 | 中文不卡视频在线 | 国产毛片在线 | 色在线视频| 亚洲国产精品va在线看 | 亚洲精品视频免费在线 | 国产二区电影 | 69精品在线观看 | 91久久久国产精品 | 麻豆系列在线观看 | 国产精品国产三级国产专区53 | 日本天天操 | 91视频啪 | 久久精品久久久久 | 日韩激情影院 | 亚洲一区免费在线 | 国产精品久久中文字幕 | 久久精品站 | 96国产精品视频 | 亚洲伦理电影在线 | 91色九色 | 天天射射天天 | 在线免费av观看 | 日韩精品在线观看av | 欧美另类v| 久久九九国产视频 | 国产精品免费久久久久久久久久中文 | 最近中文字幕高清字幕免费mv | 黄在线免费观看 | 日韩高清网站 | 免费a v视频 | 天天激情 | 欧美一区二区精品在线 | 国产免费又爽又刺激在线观看 | 日韩一级成人av | 久久精品免费播放 | 91传媒91久久久 | 娇妻呻吟一区二区三区 | 日韩69av | 亚洲午夜久久久综合37日本 | 亚洲激色 | 国产精品一区二区久久久久 | 色综合久久久久久中文网 | 成人免费视频网站 | 久久五月网 | 国产aa免费视频 | 久久亚洲欧美日韩精品专区 | 国产精品麻豆一区二区三区 | 日韩午夜剧场 | 久草资源在线 | 99这里精品 | 最近中文字幕高清字幕免费mv | 国产91精品一区二区麻豆网站 | 天天射天天射 | 97精品欧美91久久久久久 | 深夜免费福利视频 | 国产成人精品在线观看 | 日产av在线播放 | 亚洲成av人片一区二区梦乃 | 97国产在线视频 | 午夜久久影视 | 99理论片| 日日日视频| 免费网站在线观看人 | 久久久久久亚洲精品 | 亚洲日本欧美 | 色婷婷国产精品一区在线观看 | 精品视频成人 | 中文 一区二区 | 91九色免费视频 | 日本黄区免费视频观看 | 99热免费在线 | 91x色 | 国产中文字幕三区 | 草久热| 国产日韩欧美综合在线 | 国模吧一区 | 国产亚洲精品日韩在线tv黄 | 成人在线中文字幕 | av韩国在线 | 中文在线天堂资源 | 久久福利国产 | 久久久精品99 | 免费在线激情电影 | 亚州精品在线视频 | 久久8| 91av视频网 | 九九免费在线观看视频 | 亚洲精品色婷婷 | 欧美日韩中文国产 | av一级一片| 日韩精品在线观看视频 | 91最新网址 | 96av麻豆蜜桃一区二区 | 玖玖视频网| 国产精品久久一区二区无卡 | 久久99精品久久久久久秒播蜜臀 | 美女久久久久久久久久 | 国产精品免费观看久久 | 色在线网站 | 久草在线综合网 | 91精选| 成人黄色电影在线观看 | 色综合亚洲精品激情狠狠 | 免费在线看成人av | 久久经典视频 | 98超碰在线观看 | 久久黄色小说 | 玖玖视频免费在线 | 成人动图| 日韩一区二区免费视频 | 97精产国品一二三产区在线 | 国产精品久久久久四虎 | 国产成人精品av久久 | 天天se天天cao天天干 | 欧美小视频在线观看 | 国产精品一区二区吃奶在线观看 | 天天干,天天射,天天操,天天摸 | 日本韩国中文字幕 | 在线视频日韩精品 | 日韩超碰在线 | 欧美国产日韩一区二区三区 | 涩涩在线| 日韩免费视频 | 久久精品国产精品 | 中文字幕免费 | 99热精品免费观看 | 免费看一级特黄a大片 | 亚洲免费激情 | 久草网站在线观看 | 99色婷婷| 久久久亚洲精华液 | 日日日干 | 免费在线观看一级片 | 久久一区精品 | 亚洲精品国产成人av在线 | 久久综合色一综合色88 | 色婷婷精品大在线视频 | 日韩 在线a | 免费一级日韩欧美性大片 | 国产精品乱码高清在线看 | 99久久久久久国产精品 | 亚洲激情在线 | 欧美午夜a | 中文字幕av在线免费 | 欧美日在线 | 久久精品免费观看 | 99国产精品久久久久久久久久 | 天天操天天舔天天爽 | 噜噜色官网 | 亚洲国产网站 | 天天操天天干天天操天天干 | 18国产精品白浆在线观看免费 | 日本女人b| 午夜视频在线瓜伦 | 亚洲电影av在线 | 亚洲国产中文字幕在线视频综合 | 精品国产乱码久久久久久1区二区 | 91亚洲国产成人久久精品网站 | 成人中文字幕+乱码+中文字幕 | 97超级碰碰碰视频在线观看 | 国产视频丨精品|在线观看 国产精品久久久久久久久久久久午夜 | 成人性生交大片免费看中文网站 | 欧美一二三视频 | 91精品国产综合久久福利不卡 | 91丨九色丨91啦蝌蚪老版 | 婷婷丁香狠狠爱 | 99色网站 | 狠狠色丁香婷婷综合橹88 | 国产999| 国产夫妻自拍av | 伊人色综合久久天天 | 视频三区 | 欧美先锋影音 | av电影在线免费观看 | 黄免费网站| 园产精品久久久久久久7电影 | 日日操天天操夜夜操 | www日| 久艹视频在线观看 | 成人黄色小说视频 | 亚洲综合视频在线 | 欧美精品做受xxx性少妇 | 久久综合久久综合这里只有精品 | 久草视频在线新免费 | 亚洲精品在线视频播放 | 色综合久久五月 | 日本aaaa级毛片在线看 | av 在线观看 | 超碰日韩 | 69av视频在线观看 | 97在线观视频免费观看 | 精品国产免费看 | 人人澡人人澡人人 | 国产精品久久艹 | 五月婷婷黄色 | av在线免费观看不卡 | 国产高清 不卡 | 99视频国产精品 | 亚洲精品黄 | 激情视频在线观看网址 | 91豆花在线| 婷婷久久亚洲 | 日韩91av| 日日干夜夜骑 | 美女网站在线播放 | 国产永久免费观看 | 天天操天天干天天综合网 | 国际精品久久久久 | 在线中文字幕一区二区 | 婷婷在线精品视频 | 国产精品一区二区在线 | 一区二区三区四区五区在线 | 久久伊人婷婷 | 午夜精品av | 国产精品va在线 | 国产精品乱码久久久久 | 国产精品一区二区久久国产 | 九九视频这里只有精品 | 国产日韩欧美在线免费观看 | 一二三区在线 | 国产成a人亚洲精v品在线观看 | 免费人成网ww44kk44 | 国产精品成人一区二区三区 | 这里有精品在线视频 | 久色小说| 国产色女| 精品国产一区二区三区免费 | 一本—道久久a久久精品蜜桃 | 国产精品美女久久久 | 国产精品99爱 | 国产成人精品在线观看 | 91免费高清在线观看 | 丁香六月天婷婷 | 中国一区二区视频 | 久久一二区 | 在线观看不卡视频 | 在线观看蜜桃视频 | 操操操日日日 | 欧美大香线蕉线伊人久久 | 日韩,精品电影 | 99久久久国产精品免费观看 | 中文字幕 国产专区 | 人人干在线 | 免费高清在线观看成人 | 国产精品区在线观看 | 久草在线资源观看 | 黄色特一级 | 在线免费观看视频a | 精品国产理论片 | 欧美热久久 | 色婷婷av一区 | 99热播精品 | 久久中文字幕视频 | 久久久久亚洲国产精品 | 中文字幕亚洲综合久久五月天色无吗'' | 操操色| 天天综合色天天综合 | 亚洲国产福利视频 | 国产日韩欧美视频在线观看 | 国产中文字幕一区 | 在线视频国产区 | 国产精品久久久久久久久久三级 | av网址aaa| 五月婷婷视频在线观看 | 狠狠干天天射 | 欧美国产亚洲精品久久久8v | 99久久精品国产毛片 | 国产资源站| 久久免费视频在线 | 国产精品99久久久久久久久久久久 | 在线播放91 | 久久精品久久国产 | 91成人精品视频 | 亚洲综合在线观看视频 | 亚洲国产激情 | 国产日韩中文字幕 | 草久久影院 | 日韩精品在线看 | 狠狠操天天射 | 国内精品久久久久久久影视简单 | 国产成人精品一区二区三区免费 | 日韩在线精品一区 | 狠狠干我 | 亚洲黄色一级大片 | 超碰免费观看 | 日日夜夜精品免费观看 | 国产精品久久久久久久久久白浆 | 在线久热 | 国产自产高清不卡 | 国产1区2区3区在线 亚洲自拍偷拍色图 | 成年人视频在线免费 | 午夜精品久久久久久久99婷婷 | 一级黄色片在线免费观看 | 日日干av| 国产精品永久在线观看 | 婷婷免费在线视频 | 亚洲黄a | 亚洲欧美日韩国产 | 亚洲国产精品va在线 | 欧美va电影 | 久久久午夜精品福利内容 | 最新中文字幕在线资源 | 美女网色| 国产精品自产拍在线观看蜜 | 丁香婷婷综合五月 | 国产欧美综合在线观看 | 国产探花视频在线播放 | 国产乱码精品一区二区三区介绍 | 日韩综合色 | 亚洲午夜大片 | 91麻豆精品国产自产在线游戏 | 91精品国 | 蜜桃传媒一区二区 | 国内少妇自拍视频一区 | 久久影视中文字幕 | 2023亚洲精品国偷拍自产在线 | 亚洲精品午夜久久久 | 日韩亚洲国产中文字幕 | www.黄色片网站 | 久久国产热| 久久久精选 | 超碰公开在线 | 激情影音| 97夜夜澡人人爽人人免费 | 色综合天天综合 | 999久久久久久久久 69av视频在线观看 | 蜜桃视频成人在线观看 | 欧美性色19p | 国产黄色一级片 | 久操综合| 日韩在线不卡 | 在线成人性视频 | 久草在线视频免赞 | 亚洲a在线观看 | 美女免费黄视频网站 | 亚洲婷婷伊人 | 成人在线视频观看 | 亚洲 欧洲av | 婷婷丁香综合 | 国产一区二区三区午夜 | 91精品国产三级a在线观看 | 成年人免费电影 | 三上悠亚一区二区在线观看 | 亚洲一区视频在线播放 | 五月婷婷一区 | 91精品对白一区国产伦 | 成人欧美日韩国产 | 国产高清视频在线 | 黄色大片视频网站 | 免费色视频 | 日本护士撒尿xxxx18 | 91爱在线| 久久激情五月激情 | 香蕉视频在线免费 | 国产精品2019 | 亚洲成人在线免费 | 少妇bbw搡bbbb搡bbbb| 高清免费在线视频 | 日韩簧片在线观看 | 精品在线观看一区二区 | 区一区二在线 | 色视频 在线 | 国产午夜精品久久 | av片在线看 | 国产精品久久免费看 | 欧美日韩高清免费 | 最近中文字幕高清字幕在线视频 | 四月婷婷在线观看 | 天干啦夜天干天干在线线 | 国产香蕉97碰碰久久人人 | 国产视频1 | 日韩一区二区三区免费视频 | 久久精品国产亚洲精品2020 | 91在线免费看片 | 一级黄色av| 欧美精品国产精品 | 91av在线电影 | 国产成人精品一区一区一区 | 91完整版| 91视频高清免费 | 亚洲精品h | 福利视频导航网址 | 国产精品一区二区三区观看 | 国产 日韩 欧美 在线 | www.国产在线观看 | 亚洲爱av| 中文字幕一区二区三区四区在线视频 | 2018好看的中文在线观看 | 色婷婷天天干 | 最近中文字幕国语免费av | 成人教育av| 日韩免费网站 | 手机看片福利 | 日韩av电影国产 | 狠狠五月婷婷 | 久久综合免费 | 天天干婷婷 | 五月黄色| 免费观看黄 | 五月天婷亚洲天综合网鲁鲁鲁 | 99热这里只有精品久久 | 日韩美女av在线 | 青青五月天 | 欧美成人精品xxx | 免费 在线 中文 日本 | 国产亚洲免费的视频看 | 最新国产中文字幕 | av高清网站在线观看 | 深夜免费福利 | 在线精品在线 | 日韩av一卡二卡三卡 | 中文字幕九九 | 在线电影91| 人人插人人玩 | 一区在线电影 | 天天操天天操天天 | 亚洲午夜精品久久久久久久久 | 亚洲97在线 | 精品99在线视频 | 97视频在线观看网址 | 国产精品初高中精品久久 | 青春草视频在线播放 | 精品国产一区二区三区久久久久久 | 欧美成人猛片 | 国产粉嫩在线 | 国内精品视频在线播放 | av黄色国产 | 91精品亚洲影视在线观看 | 国产精品美女在线 | 久久精品站 | 久久精品久久精品久久精品 | 国产精品永久在线 | 视频一区视频二区在线观看 | 激情六月婷婷久久 | 毛片一区二区 | 中文在线免费观看 | 久久久免费精品 | 91av网址| 丁香六月网 | www亚洲国产 | 激情欧美丁香 | 久久电影色 | 啪啪午夜免费 | 亚洲更新最快 | 91精彩在线视频 | 99视频播放 | 成人天堂网 | 成人 亚洲 欧美 | 成人免费在线看片 | 欧美精品久久久久久久久老牛影院 | 九九九在线观看 | 久久久夜色 | 国产粉嫩在线 | 欧美日韩大片在线观看 | 三上悠亚一区二区在线观看 | 亚洲,国产成人av | 国产黄色精品在线 | av女优中文字幕在线观看 | 夜夜骑天天操 | 久久综合激情 | 精品视频在线免费观看 | 日日夜夜精品免费 | 麻豆传媒一区二区 | 日日爽天天操 | 开心色激情网 | 欧美另类色图 | 国产亚洲精品久 | 亚洲欧美精品一区 | 亚洲天堂精品视频在线观看 | 91av手机在线观看 | 日本精品一区二区 | 人人涩 | 久久久www成人免费精品张筱雨 | 欧美色图视频一区 | 免费一级黄色 | 欧美爽爽爽| 免费黄色在线播放 | 91亚洲网| 国产欧美中文字幕 | 五月婷色| 免费看的黄色 | 看片网站黄色 | 国产不卡精品 | 日韩毛片在线一区二区毛片 | 天天综合导航 | 亚洲精品在线免费观看视频 | 国产精品久久久久av福利动漫 | 国产一区二区视频在线 | 波多野结衣电影一区二区三区 | 国产又黄又爽无遮挡 | 欧美永久视频 | 亚洲精品乱码久久 | 在线高清一区 | 国产在线播放不卡 | 美女av免费 | 久久久网站 | 久久人人97超碰国产公开结果 | 婷婷成人在线 | 国产精品欧美久久久久三级 | 免费在线国产视频 | 色丁香婷婷 | 在线免费观看国产 | 日韩在线一二三区 | 高清av免费看| 亚洲精品成人网 | 黄色的片子 | www久久国产 | 亚洲aaa级 | 中文字幕在线有码 | 中文字幕在线观看亚洲 | 国产99久久久国产 | 色五丁香| 五月婷婷六月丁香激情 | 九九激情视频 | 亚洲午夜不卡 | 国产精品免费麻豆入口 | 精品久久久久久久久久久久久久久久久久 | 天天色天天射天天操 | 黄色影院在线观看 | 日日干美女 | 久久精品99国产精品亚洲最刺激 | 欧美日韩国产一二三区 | 激情小说网站亚洲综合网 | 日本高清久久久 | 成人超碰在线 | 国产九色在线播放九色 | 国产成人福利在线 | 区一区二区三在线观看 | 中文字幕日韩av | 久久视频免费在线 | 国产高清 不卡 | 手机成人av在线 | 久久久久久久电影 | 亚洲资源视频 | 一级做a爱片性色毛片www | 91人人人 | 婷婷综合导航 | 久久精品久久99精品久久 | 亚洲精品乱码久久久久久蜜桃动漫 | 日韩在线观看电影 | 麻豆影视网站 | 日韩二区精品 | 成人av电影在线观看 | 国色天香av | 久久久久成人免费 | 天天艹天天操 | 久久99精品久久久久久清纯直播 | 九热精品 | 高清一区二区三区 | 九九热在线播放 | 中文字幕在线免费看线人 | 99精品视频在线看 | 国偷自产中文字幕亚洲手机在线 | 欧美国产一区在线 | 国产精品免费一区二区 | 日韩高清成人在线 | 久久精品成人 | 深夜男人影院 | 国内视频在线观看 | 欧美午夜视频在线 | 日韩视频一二三区 | 国产在线观看免 | 久久精品国产亚洲精品2020 | 国产精品theporn | 一级片视频在线 | 日韩精品中文字幕在线不卡尤物 | 中文字幕a∨在线乱码免费看 | 超碰97人人干 | 91精品在线播放 | 亚洲精品videossex少妇 | 亚洲韩国一区二区三区 | 丁香影院在线 | 黄色网在线播放 | 91视频 - x99av | 国产不卡精品 | 在线观看日本高清mv视频 | 激情视频网页 | 国产成人在线一区 | 91色在线观看视频 | 中文字幕 在线看 | 日韩欧美在线一区二区 | 成年人电影免费看 | 欧美少妇bbwhd | 欧美欧美| 欧美一二区视频 | 在线观看av小说 | av成人在线播放 | 日本久久久精品视频 | 日韩成人免费电影 | 黄色免费网站下载 | 亚洲精品动漫在线 | 99免费在线视频 | 国产丝袜一区二区三区 | 欧美少妇影院 | 高清国产在线一区 | 国产视频美女 | 日韩精品91偷拍在线观看 | 久99久在线| 欧美日韩中文国产一区发布 | 免费亚洲黄色 | 久久国产精品免费视频 | 人人人爽| 久久99精品波多结衣一区 | 中文字幕日韩精品有码视频 | 91电影福利 | 九九热中文字幕 | 免费看黄色大全 | 国产精品久久久久久av | 97免费公开视频 | 日本99精品| 日韩性xxxx | 日韩网 | 在线观看av小说 | 日韩中文字幕免费在线观看 | 久久免费视频5 | 免费久久99精品国产婷婷六月 | 久久电影中文字幕视频 | 五月婷婷,六月丁香 | 99久久99久久精品免费 | 三级av在线免费观看 | 亚洲国产69 | 亚洲毛片久久 | 国产操在线| 久久在线一区 | 精品亚洲视频在线观看 | www黄免费 | 中文字幕一区二区在线观看 | 在线观看一区二区视频 | 欧美日韩3p | 在线视频日韩一区 | 黄色在线观看免费 | 午夜国产一区二区 | 午夜视频免费在线观看 | 国产日韩视频在线播放 | av 在线观看 | 精品五月天 | 最新日本中文字幕 | 国产精品一区二区三区久久久 | 亚洲片在线 | 91在线色 | 国产在线观看你懂得 | 少妇啪啪av入口 | 日韩免费一级a毛片在线播放一级 | 国产一级视频在线免费观看 | 国产高清在线免费视频 | 午夜视频在线观看网站 | 国产91对白在线播 | 国产精品午夜免费福利视频 | 免费观看视频的网站 | 亚洲欧洲久久久 | 黄色三级视频片 | 免费在线观看黄 | 欧美国产精品一区二区 | 丁香5月婷婷久久 | 日韩久久精品一区二区 | 国产精品一区二区三区免费视频 | 在线观看免费av网 | 国产精品美女网站 | 亚洲波多野结衣 | 最新av观看 | 精品久久久久久久久久岛国gif | 中文字幕色在线 | 免费成人黄色 | 高清美女视频 | 又黄又刺激视频 | 亚洲三级性片 | 中文字幕亚洲精品日韩 | 亚洲区另类春色综合小说 | 国产美女在线精品免费观看 | 手机av在线免费观看 | 青青网视频 | 国产中文字幕一区 | 最近中文国产在线视频 | 日韩av影视在线 | 在线观看av的网站 | 久草在线官网 | 依人成人综合网 | av免费观看在线 | 69久久99精品久久久久婷婷 | 麻豆视频免费入口 | 综合久久网站 | 很黄很污的视频网站 | 美女久久久久久久久久 | 日日干天天 | 日韩精品中文字幕av | 欧美一级片播放 | 国产亚洲精品成人av久久ww | 国产美女免费看 | 久久久久久久久久久高潮一区二区 | 91.dizhi永久地址最新 | 欧美精品久久久久久久久久久 | 欧美极品xxx | 综合色久| 日批视频在线观看免费 | 在线日韩av | 亚洲视频 中文字幕 | 天天干视频在线 | 97高清视频 | 成人免费观看大片 | 国产免费观看av | www.夜夜草 | 在线观看资源 | 精品少妇一区二区三区在线 | 久久国产经典视频 | 日韩视频一区二区 | 久久国产精品色婷婷 | 人人草人 | 国产69精品久久99的直播节目 | 久久久毛片 | 中文字幕乱码电影 | 国产综合在线视频 | av在线免费网站 | 韩国av免费观看 | 天天天操操操 | 超碰在线9| 色搞搞| 一级精品视频在线观看宜春院 | 国产精选在线 | 国产黑丝一区二区三区 | 天堂在线免费视频 | 国产精品一区二区麻豆 | 久久人操 | 黄色com| 亚洲狠狠婷婷 | 午夜三级在线 | 91桃色免费观看 | 久久久亚洲影院 | 国产精品色婷婷视频 | 91人人爽人人爽人人精88v | 亚洲资源在线观看 | 97在线观看视频免费 | 久久九九免费视频 | 成人网大片| 精品久久久久久电影 | 久久久久草 | 激情丁香 | 婷婷网在线 | 夜夜高潮夜夜爽国产伦精品 | 国外av在线| 蜜臀aⅴ国产精品久久久国产 | 激情黄色av | 久久精品国产一区 | 天天曰视频 | 中文字幕在线免费观看视频 | 中文字幕一二三区 | 成人va在线观看 | 国产高清专区 | 美女久久视频 | 99精品电影| www成人av | 在线观看日韩精品 | 久操久| 91av在线播放视频 | 欧美日韩不卡一区二区 | 成人在线视频一区 | 亚洲精品字幕 | 久久久在线视频 | 亚洲精品高清一区二区三区四区 | 亚洲黄色免费网站 | 日韩免费久久 | 一区二区精品国产 | 日日碰狠狠躁久久躁综合网 | 国产一区在线精品 | 中国一级片免费看 | 久久国产一二区 | 亚洲精品一区二区三区在线观看 | 国产香蕉97碰碰久久人人 | 91麻豆精品国产91久久久无需广告 | 精品在线播放 | 亚洲中字幕 | 日韩在线观看精品 | 草久中文字幕 | 91日本在线播放 | 99久久精品费精品 | 欧美综合在线视频 | 亚洲成色 | 欧洲色吧| 蜜臀aⅴ国产精品久久久国产 | 9在线观看免费高清完整版在线观看明 | 高清久久久 | 久久久香蕉视频 | 国产成a人亚洲精v品在线观看 | 一区二区三区免费看 | 天天综合网天天综合色 | 中文字幕在线观看的网站 | 成人在线免费观看网站 | 国产区在线视频 | 国产成人精品久久久 | 91精品一区二区在线观看 | 久久精品99国产精品亚洲最刺激 | 久久伊人五月天 | 国产不卡在线播放 | 欧美另类亚洲 | 国产精品18久久久久白浆 | 99久久精品免费看国产四区 | 精品婷婷 | 欧美国产亚洲精品久久久8v | 免费在线观看黄色网 | 久草视频中文在线 | 在线欧美小视频 | 婷婷深爱激情 | 欧美三人交 | 激情综合婷婷 | 玖玖精品视频 | 日韩午夜视频在线观看 | 国产精品久久久久久久久免费看 | 亚洲一级黄色大片 | 亚洲精品麻豆视频 | 久久99国产精品二区护士 | 中文字幕精品一区 | 美女精品久久 | 成人在线免费视频观看 | 日韩簧片在线观看 | 精品国产乱码久久久久 | 亚洲一区日韩精品 | 99久久久久久久久久 | 综合五月| 亚州精品天堂中文字幕 | 国内精品国产三级国产aⅴ久 | 99视频在线精品国自产拍免费观看 | 91精品视频在线免费观看 | 视频一区二区在线观看 | 国产91全国探花系列在线播放 | 91视频一8mav | 激情久久综合网 | 国产亚洲aⅴaaaaaa毛片 | 91久久精品一区二区二区 | 亚洲激情 欧美激情 | 欧美另类美少妇69xxxx | 亚洲精品综合在线观看 | 欧美极品少妇xxxxⅹ欧美极品少妇xxxx亚洲精品 | av电影中文 | 久久久久免费精品 | 亚洲精品国偷自产在线99热 | 亚洲做受高潮欧美裸体 | 成人a毛片| 香蕉视频免费在线播放 | 亚洲欧美综合精品久久成人 | 久久成人精品视频 | 亚洲精品国产综合久久 | 精品国产一区二区三区男人吃奶 | 日韩免费在线观看视频 | 亚洲成人国产 | 国产丝袜网站 | 五月婷婷一级片 | 在线视频 你懂得 | 日本在线观看中文字幕无线观看 | 在线草| 成人av免费 | av在线电影播放 | 美女国产| 丁香六月婷婷激情 | 国产一二区视频 | 黄色软件视频网站 | 激情一区二区三区欧美 | 69av视频在线观看 | 一区二区视频欧美 | www.av小说| 欧美视频国产视频 | 欧美精品网站 | 天天干天天草 | 91麻豆产精品久久久久久 | 玖玖在线播放 | 丁香五月缴情综合网 | 热久久在线视频 | 精品久久久久久国产 | 久久精品亚洲一区二区三区观看模式 | 亚洲欧美视频在线 | 国产精品久久久久国产a级 激情综合中文娱乐网 | 在线播放 一区 | 中文字幕在线视频国产 | 久久爱导航 | 国产污视频在线观看 | 欧美日韩高清在线一区 | 午夜久久网 | 最新免费av在线 | 亚洲成人黄色在线 | www黄色com | 国产一级视屏 | 伊人婷婷网 | 国产高清不卡一区二区三区 | av一级片网站 | 香蕉在线观看 | 国产一区在线视频 | 久久国产精品免费观看 | 国产精品久久久久国产a级 激情综合中文娱乐网 | 亚洲午夜久久久久久久久久久 | 亚洲精品视频在线播放 | 丁香六月天婷婷 | 欧美日韩亚洲一 | 亚洲专区中文字幕 | 成人蜜桃| 亚洲男男gaygay无套 | 91久久黄色 | 91av视频免费在线观看 | 中文字幕在线观看网站 | 蜜臀久久99精品久久久酒店新书 | 91精品国产福利在线观看 | 国精产品999国精产品视频 | 久久黄色影院 | 又黄又爽又色无遮挡免费 | 欧美日bb | 久久久久久国产精品免费 | 天天综合成人网 | 精品国模一区二区三区 | 99中文在线 | 中文字幕亚洲情99在线 | 六月色丁香 | 日韩簧片在线观看 | 人人玩人人爽 | 免费下载高清毛片 | 久久av在线 | 五月天婷婷视频 | 久久综合久久综合久久 | 成人免费看片98欧美 | 99精品国产aⅴ | 一级黄色片网站 | 日韩在线观看中文字幕 | 国产香蕉视频在线播放 | 99热 精品在线 | 成人黄色小视频 | 91麻豆精品国产自产在线游戏 | 欧美日韩免费观看一区=区三区 | 日本中文字幕免费观看 | 在线观看中文字幕av | 黄色毛片电影 | 亚洲精品国产电影 | 草久在线观看 | 欧美成年人在线观看 | 99精品黄色片免费大全 | 国产在线一线 | 国产一级片免费播放 | 久久国产精品一区二区三区四区 | 成人久久久久 | 午夜av在线免费 | 能在线观看的日韩av | 日韩有码在线观看视频 | 午夜久久福利视频 | 91成人精品一区在线播放69 | 手机成人免费视频 | 国产高清在线视频 | 精品国产一区二区三区久久 | 一本到在线| 手机在线观看国产精品 | 久久精品超碰 | 99精品久久久久久久 | 亚洲 中文 在线 精品 | 黄网站大全 | 国产精品欧美久久久久三级 | 99精品久久精品一区二区 | 69精品视频在线观看 | 免费看黄视频 | 婷婷婷国产在线视频 | 精品国产伦一区二区三区观看方式 | 亚洲精品乱码久久久久久写真 | 国产精品va在线观看入 | av在线在线| 999成人 | 99久久久久免费精品国产 | 91日韩在线视频 | 毛片区| 缴情综合网五月天 | 欧美精品视| 国产亚洲精品久久久久久网站 | 欧美成人性战久久 | 日韩免费观看一区二区三区 | 亚洲精品影院在线观看 | 国产在线第三页 | 日韩精品高清不卡 | 日韩欧美网址 | 欧美日韩国产页 | 国产视频99| 久草男人天堂 | 在线三级av | 有码中文字幕在线观看 | 成人国产精品免费观看 | 国产专区在线播放 | 午夜久久久久久久久 | 91插插视频| 免费网站观看www在线观看 | 免费人成网 | 黄色免费网站下载 | 在线观看一区二区精品 | 免费情趣视频 | 97超碰人人澡人人爱学生 | 精品国产乱码久久久久久浪潮 |