色哟哟视频在线观看-色哟哟视频在线-色哟哟欧美15最新在线-色哟哟免费在线观看-国产l精品国产亚洲区在线观看-国产l精品国产亚洲区久久

0
  • 聊天消息
  • 系統消息
  • 評論與回復
登錄后你可以
  • 下載海量資料
  • 學習在線課程
  • 觀看技術視頻
  • 寫文章/發帖/加入社區
會員中心
創作中心

完善資料讓更多小伙伴認識你,還能領取20積分哦,立即完善>

3天內不再提示

Google編程風格指南(六)

C語言專家集中營 ? 來源:未知 ? 作者:李倩 ? 2018-09-27 18:18 ? 次閱讀

9. 格式

每個人都可能有自己的代碼風格和格式, 但如果一個項目中的所有人都遵循同一風格的話, 這個項目就能更順利地進行. 每個人未必能同意下述的每一處格式規則, 而且其中的不少規則需要一定時間的適應, 但整個項目服從統一的編程風格是很重要的, 只有這樣才能讓所有人輕松地閱讀和理解代碼.

為了幫助你正確的格式化代碼, 我們寫了一個emacs 配置文件.

9.1. 行長度

總述

每一行代碼字符數不超過 80.

我們也認識到這條規則是有爭議的, 但很多已有代碼都遵照這一規則, 因此我們感覺一致性更重要.

優點

提倡該原則的人認為強迫他們調整編輯器窗口大小是很野蠻的行為. 很多人同時并排開幾個代碼窗口, 根本沒有多余的空間拉伸窗口. 大家都把窗口最大尺寸加以限定, 并且 80 列寬是傳統標準. 那么為什么要改變呢?

缺點

反對該原則的人則認為更寬的代碼行更易閱讀. 80 列的限制是上個世紀 60 年代的大型機的古板缺陷; 現代設備具有更寬的顯示屏, 可以很輕松地顯示更多代碼.

結論

80 個字符是最大值.

如果無法在不傷害易讀性的條件下進行斷行, 那么注釋行可以超過 80 個字符, 這樣可以方便復制粘貼. 例如, 帶有命令示例或 URL 的行可以超過 80 個字符.

包含長路徑的#include語句可以超出80列.

頭文件保護可以無視該原則.

9.2. 非 ASCII 字符

總述

盡量不使用非 ASCII 字符, 使用時必須使用 UTF-8 編碼.

說明

即使是英文, 也不應將用戶界面的文本硬編碼到源代碼中, 因此非 ASCII 字符應當很少被用到. 特殊情況下可以適當包含此類字符. 例如, 代碼分析外部數據文件時, 可以適當硬編碼數據文件中作為分隔符的非 ASCII 字符串; 更常見的是 (不需要本地化的) 單元測試代碼可能包含非 ASCII 字符串. 此類情況下, 應使用 UTF-8 編碼, 因為很多工具都可以理解和處理 UTF-8 編碼.

十六進制編碼也可以, 能增強可讀性的情況下尤其鼓勵 —— 比如"\xEF\xBB\xBF", 或者更簡潔地寫作u8"\uFEFF", 在 Unicode 中是零寬度 無間斷的間隔符號, 如果不用十六進制直接放在 UTF-8 格式的源文件中, 是看不到的.

(Yang.Y 注:"\xEF\xBB\xBF"通常用作 UTF-8 with BOM 編碼標記)

使用u8前綴把帶uXXXX轉義序列的字符串字面值編碼成 UTF-8. 不要用在本身就帶 UTF-8 字符的字符串字面值上, 因為如果編譯器不把源代碼識別成 UTF-8, 輸出就會出錯.

別用 C++11 的char16_t和char32_t, 它們和 UTF-8 文本沒有關系,wchar_t同理, 除非你寫的代碼要調用 Windows API, 后者廣泛使用了wchar_t.

9.3. 空格還是制表位

總述

只使用空格, 每次縮進 2 個空格.

說明

我們使用空格縮進. 不要在代碼中使用制表符. 你應該設置編輯器將制表符轉為空格.

9.4. 函數聲明與定義

總述

返回類型和函數名在同一行, 參數也盡量放在同一行, 如果放不下就對形參分行, 分行方式與函數調用一致.

說明

函數看上去像這樣:

ReturnType ClassName::FunctionName(Type par_name1, Type par_name2) { DoSomething(); ...}

如果同一行文本太多, 放不下所有參數:

ReturnType ClassName::ReallyLongFunctionName(Type par_name1, Type par_name2, Type par_name3) { DoSomething(); ...}

甚至連第一個參數都放不下:

ReturnType LongClassName::ReallyReallyReallyLongFunctionName( Type par_name1, // 4 space indent Type par_name2, Type par_name3) { DoSomething(); // 2 space indent ...}

注意以下幾點:

使用好的參數名.

只有在參數未被使用或者其用途非常明顯時, 才能省略參數名.

如果返回類型和函數名在一行放不下, 分行.

如果返回類型與函數聲明或定義分行了, 不要縮進.

左圓括號總是和函數名在同一行.

函數名和左圓括號間永遠沒有空格.

圓括號與參數間沒有空格.

左大括號總在最后一個參數同一行的末尾處, 不另起新行.

右大括號總是單獨位于函數最后一行, 或者與左大括號同一行.

右圓括號和左大括號間總是有一個空格.

所有形參應盡可能對齊.

缺省縮進為 2 個空格.

換行后的參數保持 4 個空格的縮進.

未被使用的參數, 或者根據上下文很容易看出其用途的參數, 可以省略參數名:

class Foo { public: Foo(Foo&&); Foo(const Foo&); Foo& operator=(Foo&&); Foo& operator=(const Foo&);};

未被使用的參數如果其用途不明顯的話, 在函數定義處將參數名注釋起來:

class Shape { public: virtual void Rotate(double radians) = 0;};class Circle : public Shape { public: void Rotate(double radians) override;};void Circle::Rotate(double /*radians*/) {}// 差 - 如果將來有人要實現, 很難猜出變量的作用.void Circle::Rotate(double) {}

屬性, 和展開為屬性的宏, 寫在函數聲明或定義的最前面, 即返回類型之前:

MUST_USE_RESULT bool IsOK();

9.5. Lambda 表達式

總述

Lambda 表達式對形參和函數體的格式化和其他函數一致; 捕獲列表同理, 表項用逗號隔開.

說明

若用引用捕獲, 在變量名和&之間不留空格.

int x = 0;auto add_to_x = [&x](int n) { x += n; };

短 lambda 就寫得和內聯函數一樣.

std::set blacklist = {7, 8, 9};std::vector digits = {3, 9, 1, 8, 4, 7, 1};digits.erase(std::remove_if(digits.begin(), digits.end(), [&blacklist](int i) { return blacklist.find(i) != blacklist.end(); }), digits.end());

9.6. 函數調用

總述

要么一行寫完函數調用, 要么在圓括號里對參數分行, 要么參數另起一行且縮進四格. 如果沒有其它顧慮的話, 盡可能精簡行數, 比如把多個參數適當地放在同一行里.

說明

函數調用遵循如下形式:

bool retval = DoSomething(argument1, argument2, argument3);

如果同一行放不下, 可斷為多行, 后面每一行都和第一個實參對齊, 左圓括號后和右圓括號前不要留空格:

bool retval = DoSomething(averyveryveryverylongargument1, argument2, argument3);

參數也可以放在次行, 縮進四格:

if (...) { ... ... if (...) { DoSomething( argument1, argument2, // 4 空格縮進 argument3, argument4); }

把多個參數放在同一行以減少函數調用所需的行數, 除非影響到可讀性. 有人認為把每個參數都獨立成行, 不僅更好讀, 而且方便編輯參數. 不過, 比起所謂的參數編輯, 我們更看重可讀性, 且后者比較好辦:

如果一些參數本身就是略復雜的表達式, 且降低了可讀性, 那么可以直接創建臨時變量描述該表達式, 并傳遞給函數:

int my_heuristic = scores[x] * y + bases[x];bool retval = DoSomething(my_heuristic, x, y, z);

或者放著不管, 補充上注釋:

bool retval = DoSomething(scores[x] * y + bases[x], // Score heuristic. x, y, z);

如果某參數獨立成行, 對可讀性更有幫助的話, 那也可以如此做. 參數的格式處理應當以可讀性而非其他作為最重要的原則.

此外, 如果一系列參數本身就有一定的結構, 可以酌情地按其結構來決定參數格式:

// 通過 3x3 矩陣轉換 widget.my_widget.Transform(x1, x2, x3, y1, y2, y3, z1, z2, z3);

9.7. 列表初始化格式

總述

您平時怎么格式化函數調用, 就怎么格式化列表初始化.

說明

如果列表初始化伴隨著名字, 比如類型或變量名, 格式化時將將名字視作函數調用名,{}視作函數調用的括號. 如果沒有名字, 就視作名字長度為零.

// 一行列表初始化示范.return {foo, bar};functioncall({foo, bar});pair p{foo, bar};// 當不得不斷行時.SomeFunction( {"assume a zero-length name before {"}, // 假設在 { 前有長度為零的名字. some_other_function_parameter);SomeType variable{ some, other, values, {"assume a zero-length name before {"}, // 假設在 { 前有長度為零的名字. SomeOtherType{ "Very long string requiring the surrounding breaks.", // 非常長的字符串, 前后都需要斷行. some, other values}, SomeOtherType{"Slightly shorter string", // 稍短的字符串. some, other, values}};SomeType variable{ "This is too long to fit all in one line"}; // 字符串過長, 因此無法放在同一行.MyType m = { // 注意了, 您可以在 { 前斷行. superlongvariablename1, superlongvariablename2, {short, interior, list}, {interiorwrappinglist, interiorwrappinglist2}};

9.9. 條件語句

總述

傾向于不在圓括號內使用空格. 關鍵字if和else另起一行.

說明

對基本條件語句有兩種可以接受的格式. 一種在圓括號和條件之間有空格, 另一種沒有.

最常見的是沒有空格的格式. 哪一種都可以, 最重要的是保持一致. 如果你是在修改一個文件, 參考當前已有格式. 如果是寫新的代碼, 參考目錄下或項目中其它文件. 還在猶豫的話, 就不要加空格了.

if (condition) { // 圓括號里沒有空格. ... // 2 空格縮進.} else if (...) { // else 與 if 的右括號同一行. ...} else { ...}

如果你更喜歡在圓括號內部加空格:

if ( condition ) { // 圓括號與空格緊鄰 - 不常見 ... // 2 空格縮進.} else { // else 與 if 的右括號同一行. ...}

注意所有情況下if和左圓括號間都有個空格. 右圓括號和左大括號之間也要有個空格:

if(condition) // 差 - IF 后面沒空格.if (condition){ // 差 - { 前面沒空格.if(condition){ // 變本加厲地差.if (condition) { // 好 - IF 和 { 都與空格緊鄰.

如果能增強可讀性, 簡短的條件語句允許寫在同一行. 只有當語句簡單并且沒有使用else子句時使用:

if (x == kFoo) return new Foo();if (x == kBar) return new Bar();

如果語句有else分支則不允許:

// 不允許 - 當有 ELSE 分支時 IF 塊卻寫在同一行if (x) DoThis();else DoThat();

通常, 單行語句不需要使用大括號, 如果你喜歡用也沒問題; 復雜的條件或循環語句用大括號可讀性會更好. 也有一些項目要求if必須總是使用大括號:

if (condition) DoSomething(); // 2 空格縮進.if (condition) { DoSomething(); // 2 空格縮進.}

但如果語句中某個if-else分支使用了大括號的話, 其它分支也必須使用:

// 不可以這樣子 - IF 有大括號 ELSE 卻沒有.if (condition) { foo;} else bar;// 不可以這樣子 - ELSE 有大括號 IF 卻沒有.if (condition) foo;else { bar;}// 只要其中一個分支用了大括號, 兩個分支都要用上大括號.if (condition) { foo;} else { bar;}

9.9. 循環和開關選擇語句

總述

switch語句可以使用大括號分段, 以表明 cases 之間不是連在一起的. 在單語句循環里, 括號可用可不用. 空循環體應使用{}或continue.

說明

switch語句中的case塊可以使用大括號也可以不用, 取決于你的個人喜好. 如果用的話, 要按照下文所述的方法.

如果有不滿足case條件的枚舉值,switch應該總是包含一個default匹配 (如果有輸入值沒有 case 去處理, 編譯器將給出 warning). 如果default應該永遠執行不到, 簡單的加條assert:

switch (var) { case 0: { // 2 空格縮進 ... // 4 空格縮進 break; } case 1: { ... break; } default: { assert(false); }}

在單語句循環里, 括號可用可不用:

for (int i = 0; i < kSomeNumber; ++i) printf("I love you\n");for (int i = 0; i < kSomeNumber; ++i) { printf("I take it back\n");}

空循環體應使用{}或continue, 而不是一個簡單的分號.

while (condition) { // 反復循環直到條件失效.}for (int i = 0; i < kSomeNumber; ++i) {} // 可 - 空循環體.while (condition) continue; // 可 - contunue 表明沒有邏輯.while (condition); // 差 - 看起來僅僅只是 while/loop 的部分之一.

9.10. 指針和引用表達式

總述

句點或箭頭前后不要有空格. 指針/地址操作符 (*,&) 之后不能有空格.

說明

下面是指針和引用表達式的正確使用范例:

x = *p;p = &x;x = r.y;x = r->y;

注意:

在訪問成員時, 句點或箭頭前后沒有空格.

指針操作符*或&后沒有空格.

在聲明指針變量或參數時, 星號與類型或變量名緊挨都可以:

// 好, 空格前置.char *c;const string &str;// 好, 空格后置.char* c;const string& str;int x, *y; // 不允許 - 在多重聲明中不能使用 & 或 *char * c; // 差 - * 兩邊都有空格const string & str; // 差 - & 兩邊都有空格.

在單個文件內要保持風格一致, 所以, 如果是修改現有文件, 要遵照該文件的風格.

9.11. 布爾表達式

總述

如果一個布爾表達式超過標準行寬, 斷行方式要統一一下.

說明

下例中, 邏輯與 (&&) 操作符總位于行尾:

if (this_one_thing > this_other_thing && a_third_thing == a_fourth_thing && yet_another && last_one) { ...}

注意, 上例的邏輯與 (&&) 操作符均位于行尾. 這個格式在 Google 里很常見, 雖然把所有操作符放在開頭也可以. 可以考慮額外插入圓括號, 合理使用的話對增強可讀性是很有幫助的. 此外, 直接用符號形式的操作符, 比如&&和~, 不要用詞語形式的and和compl.

9.12. 函數返回值

總述

不要在return表達式里加上非必須的圓括號.

說明

只有在寫x=expr要加上括號的時候才在returnexpr;里使用括號.

return result; // 返回值很簡單, 沒有圓括號.// 可以用圓括號把復雜表達式圈起來, 改善可讀性.return (some_long_condition && another_condition);return (value); // 畢竟您從來不會寫 var = (value);return(result); // return 可不是函數!

9.13. 變量及數組初始化

總述

用=,()和{}均可.

說明

您可以用=,()和{}, 以下的例子都是正確的:

int x = 3;int x(3);int x{3};string name("Some Name");string name = "Some Name";string name{"Some Name"};

請務必小心列表初始化{...}用std::initializer_list構造函數初始化出的類型. 非空列表初始化就會優先調用std::initializer_list, 不過空列表初始化除外, 后者原則上會調用默認構造函數. 為了強制禁用std::initializer_list構造函數, 請改用括號.

vector v(100, 1); // 內容為 100 個 1 的向量.vector v{100, 1}; // 內容為 100 和 1 的向量.

此外, 列表初始化不允許整型類型的四舍五入, 這可以用來避免一些類型上的編程失誤.

int pi(3.14); // 好 - pi == 3.int pi{3.14}; // 編譯錯誤: 縮窄轉換.

9.14. 預處理指令

總述

預處理指令不要縮進, 從行首開始.

說明

即使預處理指令位于縮進代碼塊中, 指令也應從行首開始.

// 好 - 指令從行首開始 if (lopsided_score) {#if DISASTER_PENDING // 正確 - 從行首開始 DropEverything();# if NOTIFY // 非必要 - # 后跟空格 NotifyClient();# endif#endif BackToNormal(); }// 差 - 指令縮進 if (lopsided_score) { #if DISASTER_PENDING // 差 - "#if" 應該放在行開頭 DropEverything(); #endif // 差 - "#endif" 不要縮進 BackToNormal(); }

9.15. 類格式

總述

訪問控制塊的聲明依次序是public:,protected:,private:, 每個都縮進 1 個空格.

說明

類聲明 (下面的代碼中缺少注釋, 參考類注釋) 的基本格式如下:

class MyClass : public OtherClass { public: // 注意有一個空格的縮進 MyClass(); // 標準的兩空格縮進 explicit MyClass(int var); ~MyClass() {} void SomeFunction(); void SomeFunctionThatDoesNothing() { } void set_some_var(int var) { some_var_ = var; } int some_var() const { return some_var_; } private: bool SomeInternalFunction(); int some_var_; int some_other_var_;};

注意事項:

所有基類名應在 80 列限制下盡量與子類名放在同一行.

關鍵詞public:,protected:,private:要縮進 1 個空格.

除第一個關鍵詞 (一般是public) 外, 其他關鍵詞前要空一行. 如果類比較小的話也可以不空.

這些關鍵詞后不要保留空行.

public放在最前面, 然后是protected, 最后是private.

關于聲明順序的規則請參考聲明順序一節.

9.16. 構造函數初始值列表

總述

構造函數初始化列表放在同一行或按四格縮進并排多行.

說明

下面兩種初始值列表方式都可以接受:

// 如果所有變量能放在同一行:MyClass::MyClass(int var) : some_var_(var) { DoSomething();}// 如果不能放在同一行,// 必須置于冒號后, 并縮進 4 個空格MyClass::MyClass(int var) : some_var_(var), some_other_var_(var + 1) { DoSomething();}// 如果初始化列表需要置于多行, 將每一個成員放在單獨的一行// 并逐行對齊MyClass::MyClass(int var) : some_var_(var), // 4 space indent some_other_var_(var + 1) { // lined up DoSomething();}// 右大括號 } 可以和左大括號 { 放在同一行// 如果這樣做合適的話MyClass::MyClass(int var) : some_var_(var) {}

9.17. 命名空間格式化

總述

命名空間內容不縮進.

說明

命名空間不要增加額外的縮進層次, 例如:

namespace {void foo() { // 正確. 命名空間內沒有額外的縮進. ...}} // namespace

不要在命名空間內縮進:

namespace { // 錯, 縮進多余了. void foo() { ... }} // namespace

聲明嵌套命名空間時, 每個命名空間都獨立成行.

namespace foo {namespace bar {

9.19. 水平留白

總述

水平留白的使用根據在代碼中的位置決定. 永遠不要在行尾添加沒意義的留白.

說明

通用

void f(bool b) { // 左大括號前總是有空格. ...int i = 0; // 分號前不加空格.// 列表初始化中大括號內的空格是可選的.// 如果加了空格, 那么兩邊都要加上.int x[] = { 0 };int x[] = {0};// 繼承與初始化列表中的冒號前后恒有空格.class Foo : public Bar { public: // 對于單行函數的實現, 在大括號內加上空格 // 然后是函數實現 Foo(int b) : Bar(), baz_(b) {} // 大括號里面是空的話, 不加空格. void Reset() { baz_ = 0; } // 用括號把大括號與實現分開. ...

添加冗余的留白會給其他人編輯時造成額外負擔. 因此, 行尾不要留空格. 如果確定一行代碼已經修改完畢, 將多余的空格去掉; 或者在專門清理空格時去掉(尤其是在沒有其他人在處理這件事的時候). (Yang.Y 注: 現在大部分代碼編輯器稍加設置后, 都支持自動刪除行首/行尾空格, 如果不支持, 考慮換一款編輯器或 IDE)

循環和條件語句

if (b) { // if 條件語句和循環語句關鍵字后均有空格.} else { // else 前后有空格.}while (test) {} // 圓括號內部不緊鄰空格.switch (i) {for (int i = 0; i < 5; ++i) {switch ( i ) { // 循環和條件語句的圓括號里可以與空格緊鄰.if ( test ) { // 圓括號, 但這很少見. 總之要一致.for ( int i = 0; i < 5; ++i ) {for ( ; i < 5 ; ++i) { // 循環里內 ; 后恒有空格, ; ?前可以加個空格.switch (i) { case 1: // switch case 的冒號前無空格. ... case 2: break; // 如果冒號有代碼, 加個空格.

操作符

// 賦值運算符前后總是有空格.x = 0;// 其它二元操作符也前后恒有空格, 不過對于表達式的子式可以不加空格.// 圓括號內部沒有緊鄰空格.v = w * x + y / z;v = w*x + y/z;v = w * (x + z);// 在參數和一元操作符之間不加空格.x = -5;++x;if (x && !y) ...

模板和轉換

// 尖括號(< and >) 不與空格緊鄰, < 前沒有空格, > 和 ( 之間也沒有.vector x;y = static_cast(x);// 在類型與指針操作符之間留空格也可以, 但要保持一致.vector x;

9.19. 垂直留白

總述

垂直留白越少越好.

說明

這不僅僅是規則而是原則問題了: 不在萬不得已, 不要使用空行. 尤其是: 兩個函數定義之間的空行不要超過 2 行, 函數體首尾不要留空行, 函數體中也不要隨意添加空行.

基本原則是: 同一屏可以顯示的代碼越多, 越容易理解程序的控制流. 當然, 過于密集的代碼塊和過于疏松的代碼塊同樣難看, 這取決于你的判斷. 但通常是垂直留白越少越好.

下面的規則可以讓加入的空行更有效:

函數體內開頭或結尾的空行可讀性微乎其微.

在多重 if-else 塊里加空行或許有點可讀性.

譯者 (YuleFox) 筆記

對于代碼格式, 因人, 系統而異各有優缺點, 但同一個項目中遵循同一標準還是有必要的;

行寬原則上不超過 80 列, 把 22 寸的顯示屏都占完, 怎么也說不過去;

盡量不使用非 ASCII 字符, 如果使用的話, 參考 UTF-8 格式 (尤其是 UNIX/Linux 下, Windows 下可以考慮寬字符), 盡量不將字符串常量耦合到代碼中, 比如獨立出資源文件, 這不僅僅是風格問題了;

UNIX/Linux 下無條件使用空格, MSVC 的話使用 Tab 也無可厚非;

函數參數, 邏輯條件, 初始化列表: 要么所有參數和函數名放在同一行, 要么所有參數并排分行;

除函數定義的左大括號可以置于行首外, 包括函數/類/結構體/枚舉聲明, 各種語句的左大括號置于行尾, 所有右大括號獨立成行;

./->操作符前后不留空格,*/&不要前后都留, 一個就可, 靠左靠右依各人喜好;

預處理指令/命名空間不使用額外縮進, 類/結構體/枚舉/函數/語句使用縮進;

初始化用=還是()依個人喜好, 統一就好;

return不要加();

水平/垂直留白不要濫用, 怎么易讀怎么來.

關于 UNIX/Linux 風格為什么要把左大括號置于行尾 (.cc文件的函數實現處, 左大括號位于行首), 我的理解是代碼看上去比較簡約, 想想行首除了函數體被一對大括號封在一起之外, 只有右大括號的代碼看上去確實也舒服; Windows 風格將左大括號置于行首的優點是匹配情況一目了然.

譯者(acgtyrant)筆記

80 行限制事實上有助于避免代碼可讀性失控, 比如超多重嵌套塊, 超多重函數調用等等.

Linux 上設置好了 Locale 就幾乎一勞永逸設置好所有開發環境的編碼, 不像奇葩的 Windows.

Google 強調有一對 if-else 時, 不論有沒有嵌套, 都要有大括號. Apple 正好有栽過跟頭.

其實我主張指針/地址操作符與變量名緊鄰,int*a,bvsint*a,b, 新手會誤以為前者的b是int*變量, 但后者就不一樣了, 高下立判.

在這風格指南里我才剛知道 C++ 原來還有所謂的Alternative operator representations, 大概沒人用吧.

注意構造函數初始值列表(Constructer Initializer List)與列表初始化(Initializer List)是兩碼事, 我就差點混淆了它們的翻譯.

事實上, 如果您熟悉英語本身的書寫規則, 就會發現該風格指南在格式上的規定與英語語法相當一脈相承. 比如普通標點符號和單詞后面還有文本的話, 總會留一個空格; 特殊符號與單詞之間就不用留了, 比如if(true)中的圓括號與true.

本風格指南沒有明確規定 void 函數里要不要用 return 語句, 不過就 Google 開源項目 leveldb 并沒有寫; 此外從Is a blank return statement at the end of a function whos return type is void necessary?來看,return;比return;更約定俗成(事實上 cpplint 會對后者報錯, 指出分號前有多余的空格), 且可用來提前跳出函數棧.

10. 規則特例

前面說明的編程習慣基本都是強制性的. 但所有優秀的規則都允許例外, 這里就是探討這些特例.

10.1. 現有不合規范的代碼

總述

對于現有不符合既定編程風格的代碼可以網開一面.

說明

當你修改使用其他風格的代碼時, 為了與代碼原有風格保持一致可以不使用本指南約定. 如果不放心, 可以與代碼原作者或現在的負責人員商討. 記住,一致性也包括原有的一致性.

10.2. Windows 代碼

總述

Windows 程序員有自己的編程習慣, 主要源于 Windows 頭文件和其它 Microsoft 代碼. 我們希望任何人都可以順利讀懂你的代碼, 所以針對所有平臺的 C++ 編程只給出一個單獨的指南.

說明

如果你習慣使用 Windows 編碼風格, 這兒有必要重申一下某些你可能會忘記的指南:

不要使用匈牙利命名法 (比如把整型變量命名成iNum). 使用 Google 命名約定, 包括對源文件使用.cc擴展名.

Windows 定義了很多原生類型的同義詞 (YuleFox 注: 這一點, 我也很反感), 如DWORD,HANDLE等等. 在調用 Windows API 時這是完全可以接受甚至鼓勵的. 即使如此, 還是盡量使用原有的 C++ 類型, 例如使用constTCHAR*而不是LPCTSTR.

使用 Microsoft Visual C++ 進行編譯時, 將警告級別設置為 3 或更高, 并將所有警告(warnings)當作錯誤(errors)處理.

不要使用#pragmaonce; 而應該使用 Google 的頭文件保護規則. 頭文件保護的路徑應該相對于項目根目錄 (Yang.Y 注: 如#ifndefSRC_DIR_BAR_H_, 參考#define 保護一節).

除非萬不得已, 不要使用任何非標準的擴展, 如#pragma和__declspec. 使用__declspec(dllimport)和__declspec(dllexport)是允許的, 但必須通過宏來使用, 比如DLLIMPORT和DLLEXPORT, 這樣其他人在分享使用這些代碼時可以很容易地禁用這些擴展.

然而, 在 Windows 上仍然有一些我們偶爾需要違反的規則:

通常我們禁止使用多重繼承, 但在使用 COM 和 ATL/WTL 類時可以使用多重繼承. 為了實現 COM 或 ATL/WTL 類/接口, 你可能不得不使用多重實現繼承.

雖然代碼中不應該使用異常, 但是在 ATL 和部分 STL(包括 Visual C++ 的 STL) 中異常被廣泛使用. 使用 ATL 時, 應定義_ATL_NO_EXCEPTIONS以禁用異常. 你需要研究一下是否能夠禁用 STL 的異常, 如果無法禁用, 可以啟用編譯器異常. (注意這只是為了編譯 STL, 自己的代碼里仍然不應當包含異常處理).

通常為了利用頭文件預編譯, 每個每個源文件的開頭都會包含一個名為StdAfx.h或precompile.h的文件. 為了使代碼方便與其他項目共享, 請避免顯式包含此文件 (除了在precompile.cc中), 使用/FI編譯器選項以自動包含該文件.

資源頭文件通常命名為resource.h且只包含宏, 這一文件不需要遵守本風格指南.

11. 結束語

運用常識和判斷力, 并且保持一致.

編輯代碼時, 花點時間看看項目中的其它代碼, 并熟悉其風格. 如果其它代碼中if語句使用空格, 那么你也要使用. 如果其中的注釋用星號 (*) 圍成一個盒子狀, 那么你同樣要這么做.

風格指南的重點在于提供一個通用的編程規范, 這樣大家可以把精力集中在實現內容而不是表現形式上. 我們展示的是一個總體的的風格規范, 但局部風格也很重要, 如果你在一個文件中新加的代碼和原有代碼風格相去甚遠, 這就破壞了文件本身的整體美觀, 也讓打亂讀者在閱讀代碼時的節奏, 所以要盡量避免.

好了, 關于編碼風格寫的夠多了; 代碼本身才更有趣. 盡情享受吧!

聲明:本文內容及配圖由入駐作者撰寫或者入駐合作網站授權轉載。文章觀點僅代表作者本人,不代表電子發燒友網立場。文章及其配圖僅供工程師學習之用,如有內容侵權或者其他違規問題,請聯系本站處理。 舉報投訴
  • Google
    +關注

    關注

    5

    文章

    1772

    瀏覽量

    57720
  • 編程
    +關注

    關注

    88

    文章

    3637

    瀏覽量

    93908
  • C++
    C++
    +關注

    關注

    22

    文章

    2114

    瀏覽量

    73792

原文標題:Google C++ 編程規范 - 6

文章出處:【微信號:C_Expert,微信公眾號:C語言專家集中營】歡迎添加關注!文章轉載請注明出處。

收藏 人收藏

    評論

    相關推薦

    MATLAB 編程風格指南

    thebeginning.”(良好的寫作規范的程序比糟糕的寫作規范的要好,因為他們具有較少的錯誤、易于調試與修改,因此,從一開始就考慮風格是很重要的)。本指南列舉的MATLAB 代碼編寫的建議在
    發表于 09-22 16:19

    Google C++編程指南

    Google C++編程指南目標:增強代碼一致性,創建通用的、必需的習慣用語和模式可以使代碼更加容易理解C++是一門包含大量高級特性的巨型語言,某些情況下,我們會限制甚至禁止使用某些特性使代碼簡化
    發表于 11-29 09:15

    MATLAB編程風格指南

    有關 MATLAB代碼的建議通常強調的是效率,譬如說有關“不要用循環”等的建議,本指南與之不同。本指南主要考慮的是代碼(格式)的正確性、清晰性與通用性。本指南的目的在
    發表于 07-18 10:54 ?0次下載

    linux內核C語言的編程風格

    linux 內核C語言的編程風格
    發表于 09-26 14:22 ?0次下載

    Google編程風格指南(一)

    使代碼易于管理的方法之一是加強代碼一致性. 讓任何程序員都可以快速讀懂你的代碼這點非常重要. 保持統一編程風格并遵守約定意味著可以很容易根據 “模式匹配” 規則來推斷各種標識符的含義. 創建通用
    的頭像 發表于 09-27 17:57 ?3150次閱讀

    Google編程風格指南(二)

    鼓勵在 .cc 文件內使用匿名命名空間或 static 聲明. 使用具名的命名空間時, 其名稱可基于項目名或相對路徑. 禁止使用 using 指示(using-directive)。禁止使用內聯命名空間(inline namespace)。
    的頭像 發表于 09-27 18:01 ?2609次閱讀

    Google編程風格指南(三)

    C/C++ 中的函數參數或者是函數的輸入, 或者是函數的輸出, 或兼而有之. 輸入參數通常是值參或 const 引用, 輸出參數或輸入/輸出參數則一般為非 const 指針. 在排列參數順序時, 將所有的輸入參數置于輸出參數之前. 特別要注意, 在加入新參數時不要因為它們是新參數就置于參數列表最后, 而是仍然要按照前述的規則, 即將新的輸入參數也置于輸出參數之前.
    的頭像 發表于 09-27 18:06 ?2568次閱讀

    Google編程風格指南(四)

    用于定義移動構造函數 (使用類的右值引用進行構造的函數) 使得移動一個值而非拷貝之成為可能. 例如, 如果 v1 是一個 vector, 則 auto v2(std::move(v1)) 將很可能不再進行大量的數據復制而只是簡單地進行指針操作, 在某些情況下這將帶來大幅度的性能提升.
    的頭像 發表于 09-27 18:08 ?2903次閱讀

    Google編程風格指南(五)

    所有具有靜態存儲類型的變量 (例如靜態變量或全局變量, 參見 存儲類型) 都應當以此方式命名. 對于其他存儲類型的變量, 如自動變量等, 這條規則是可選的. 如果不采用這條規則, 就按照一般的變量命名規則.
    的頭像 發表于 09-27 18:15 ?2484次閱讀

    Google C++編程風格指南PDF版免費下載

    Google的項目大多使用C++開發。每一個C++程序員也都知道, C++具有很多強大的語言特性,但這種強大不可避免的導致它的復雜,而復雜性會使得代碼更容易出現bug.難于閱讀和維護。
    發表于 03-06 08:00 ?0次下載
    <b class='flag-5'>Google</b> C++<b class='flag-5'>編程</b><b class='flag-5'>風格</b><b class='flag-5'>指南</b>PDF版免費下載

    Google C++編程風格指南PDF電子書免費下載

    Google 的開源項目大多使用 C++開發。每一個 C++程序員也都知道,C++具有很多強大的語言特性,但這種強大不可避免的導致它的復雜,這種復雜會使得代碼更易于出現 bug、難于閱讀和維護。本
    發表于 12-12 08:00 ?1次下載
    <b class='flag-5'>Google</b> C++<b class='flag-5'>編程</b><b class='flag-5'>風格</b><b class='flag-5'>指南</b>PDF電子書免費下載

    Verilog HIDL的RTL設計風格指南資源下載

    Verilog HIDL的RTL設計風格指南資源下載
    發表于 04-13 10:09 ?9次下載

    西門子S7-1200和S7-1500編程風格指南

    西門子S7-1200和S7-1500編程風格指南分享
    發表于 08-17 17:30 ?21次下載

    Google 應用出海計劃 | 指南針 第期強勢回歸啟動報名!

    原文標題:Google 應用出海計劃 | 指南針 第期強勢回歸啟動報名! 文章出處:【微信公眾號:谷歌開發者】歡迎添加關注!文章轉載請注明出處。
    的頭像 發表于 04-21 19:15 ?1776次閱讀
    <b class='flag-5'>Google</b> 應用出海計劃 | <b class='flag-5'>指南</b>針 第<b class='flag-5'>六</b>期強勢回歸啟動報名!

    Google Python代碼風格指南

    1 背景 Python是谷歌主要使用的動態語言,本風格指導列舉了使用Python編程時應該做和不該做的事項(dos nothing on first line # 縮進4個空格,首行括號后無內容
    的頭像 發表于 11-03 10:20 ?3121次閱讀
    主站蜘蛛池模板: 亚洲免费视频观看 | 看美女大腿中间的部分 | 木凡的天空在线收听 | 欧美日韩一级黄色片 | 奶头好翘是不是想要了 | 久久妇女高潮几次MBA | 短篇合集纯肉高H深陷骚 | 亚洲精品第一综合99久久 | 日本女人下面毛茸茸 | 亚洲精品无码不卡在线播HE | 久久精品热只有精品 | 国内精品欧美久久精品 | 久久这里只有是精品23 | 牢记永久免费网址 | 精品午夜视频 | 国产精品96久久久久久AV不卡 | 亚洲AV久久久久久久无码 | 高h全肉图| 青青草原伊人 | 久久婷婷五月免费综合色啪 | 曰本女人牲交视频免费 | 亚洲视频91 | 亚洲黄色网页 | JIZJIZJIZ 日本老师水多 | 超碰97人人做人人爱少妇 | 忘忧草在线社区WWW日本直播 | 国产精品亚洲欧美一区麻豆 | 国产成人综合在线视频 | 国产精品AV色欲蜜臀在线 | 亚洲国产系列一区二区三区 | 秋霞伦理手机在线看片 | 大胸美女裸身色诱网站 | 92看看福利午夜影院 | 正在播放国产尾随丝袜美女 | 碰超成人在线公开免费视频 | 猫咪最新破解版下载 | 免费中文字幕视频 | 亚洲精品国产第一区第二区 | 丝袜美女被艹 | 动漫美女被吸奶 | 超碰视频在线观看 |