-
Notifications
You must be signed in to change notification settings - Fork 0
Coding Style
jay.chang edited this page Apr 24, 2019
·
2 revisions
一個團隊的編程藝術是非常重要的,以下是我整理歸納的c++ Coding Style,關於原因並不多做說明。
[TOC]
- All header files should have #define guards to prevent multiple inclusion.
-
#pragma once
: Don't duplicate the head files -
#ifndef
: The format of the symbol name should bePROJECT_<PATH_>FILENAME_H
#ifndef FOO_BAR_BAZ_H #define FOO_BAR_BAZ_H ... #endif // FOO_BAR_BAZ_H
-
- Include statements should be sorted and grouped. Sorted by their hierarchical position. Leave an empty line between groups of include statements.[^1] 由上層至底層
- Local headers of the project
- Imported third party headers
- Standard library headers (STL or cstdlib)
- System headers
#include "foo.h" //Current foo.cpp head file #include "base/basictypes.h" //Local headers of the project #include "vendor/json11/json11.hpp" //Imported third party headers #include <thread> //STL #include <vector> #include <sys/types.h> //System headers #include <unistd.h>
- 在 header 檔不使用using namespace,在實作層面才為了便利使用。
- 可簡化namespace BP = boost::process;
- 當全域變數(或函式)不想被其他檔案引用和修改時,使用unnamd namespace (anonymous namespace) 來達到scope in one file only,static在c++修飾的東西不同,意思也不盡相同,容易混淆。[^2]
[^1]: Directives Matter, What the correct way to order, C/C++ include header file order [^2]: http://archerworks.blogspot.com/2010/07/cstatic.html
- 區域變數必全小寫或第一單字小寫開頭: index, salary, is_file_open, isFileOpen
- 全域變數加上g開頭: gTime
- 全大寫:常數、eunm成員、全域函式、巨集
- class, enum, 檔案名稱單字首大寫: CamelCase, Account, TestCase, UIMain.cpp, UI_Main.cpp
- enum為複數:Angles, Corners
- class成員變數:
- private, protect加上m開頭: mName
- public: PhoneBook
- class成員函式: 動詞(+名詞),純名詞為取得封裝成員。
- private, protect首字母必小寫: openFile(), save()
- public: OpenFile(), Name()=GetName()
- 指標變數:
- 區域變數指標: pindex, p_index, pIndex
- class成員變數指標: mIndexPtr, mNamePtr, PhoneBookPtr
- 全域變數指標: gTimePtr
ps: 大寫感覺不會變動,範圍大;小寫感覺容易變動,範圍小。
- 其他一律不使用前綴、後綴。物件類別提示不算入前綴的一種,可看需求使用: ::Button BTN_SAVE, mBtnSave, gBtnSave, BtnSave, btnSave ::Label LB_NAME, mLbName, LbName, lbName
- Avoid Leading Underscores: 以 underscore 起頭的 identifier 都是保留給 compiler/standard library 實作任意使用的。
- 宣告指標(raw pointer)時,型態+空白+*+空白+名稱,若未立即初始化,務必初始化為nullptr。:
int * p = nullptr; //明確宣告為指標位址變數
int * p1 = nullptr, * p2 = nullptr;
- 使用時,*+名稱為使用內容:
p=&value; //位址變數
*p=0; //內容
- 刪除指標之後,以下情況請一律設定為nullptr
- 後面流程非接右大括號(})
- 接右大括號(})但宣告為非local scope指標
bool foo(Object * pObject, Object * qObject)
{
... new ...
delete pObject;
pObject = nullptr;
... new ...
... new ...
delete qObject;
delete gClient;
gClient = nullptr;
return true;
}
加入適當的程式註解,依據Doxygen的定義,以方便輸出Document以及維護。
標頭檔為必須,其他可視情況。
/**********************************************************************//**
* @file ModemDefine.h
* @brief Definition of Modem Service module
* @date 22 March 2009
* @version 1.3.11
*
* Here are definitons all up and down layers need.
*
* @note Basically you do not need to modify this file.
* Only if on different platform, modify the definition IN_platform.
*************************************************************************/
非必要
#define MODEM_SEND_BUFFER DEVICE_OUT_BUFFER ///< must be the same with ComPort
#define MODEM_RECV_BUFFER DEVICE_IN_BUFFER ///< must be the same with ComPort
結構(struct)、列舉(enum)為非必要
/**
* @brief Design to be a libiary of the general Modem
*
* Directly use the Modem class or inherit it
* to creat your own modem, like Modem1xxx.
*
* If on different platform, modify the definition in ModemDefine.h.
*/
class Modem
{
....
}
非必要
struct PhonebookEntry
{
std::string index; ///< 1
std::string name; ///< Tina
std::string number; ///< 0912345678
std::string email;
};
必要
/**
* @brief Change facility password.
*
* Send an AT command "AT+CPWD" series.
* <pre>
* RSSI dbm
* 0 -113 dBm or less
* 1 -111 dBm
* 2..30 -109... -53 dBm (x dbm = -109 + ( RSSI - 2 ) * 2)
* 31 -51 dBm or greater
* 99 not known or not detectable
* </pre>
* @param[in] fac A facility contained by FacSupportedChangePW
* @param[in] oldpw An old password string
* @param[in] newpw A new password string
* @param[out] p_atrsp A response string point
*
* @retval MS_OK Success
* @retval CP_INVALID_HANDLE Invalid com port handle
* @retval CP_FAIL Using com port Fail
*
* @note If response string size > MODEM_RECV_BUFFERSIZE, it only recives
* MODEM_RECV_BUFFERSIZE. Then you need to reciver the remainder again.
* @remarks To change password needs to follow the flow states:
* @code
* at+clck="sc",1,"0000" <--already done(enable) in UI
* OK
* at+cpin="0000" <--even correct pw will error, so not to do, maybe hardware bug
* OK
* at+cpwd="sc","0000","1111"
* OK
* @endcode
*
* @bug ATE1 will also set at+cmee=0, is it a at command bug?
* @todo Fix the bug.
*/
int Modem::ChangeFacilityPW(FacSupportedChangePW fac, string oldpw, string newpw, char * p_atrsp)
{
...
}
參數較少可使用@a風格簡化
/**
* Copies bytes from a source memory area to a destination memory area,
* where both areas may not overlap.
* @param[out] dest The memory area to copy to.
* @param[in] src The memory area to copy from.
* @param[in] n The number of bytes to copy.
*/
/**
* Copies @a n bytes from a source memory area @a src to a destination memory
* area @a dest, where both areas may not overlap.
*/
- 使用left hand comparisons: if(365==days)
- 使用空行分開程式碼的邏輯區塊
回傳值為bool型態時,函式名稱以Is開頭:
bool IsFun(){}
if(!IsFun()){}
其他函式名稱不可以Is開頭。
- 盡量在Class內管理資源配置,達到自動式資源釋放,避免新增物件後因Exception而中斷資源釋放。(意即RAII)
- 盡量用 assert 來確認上層呼叫下層時給的參數(環境)保證。保障安全性,也省去Release重複檢查。Release時出現這種情況就已經是意外了,以exception由上(上)層處理。
void test(const char* str)
{
assert(NULL != str);
size_t len = strlen(str);
// ...
}
- 以物件方式來管理資源,達到RAII,exception safe的程式,故配置指標時,務必使用Smart pointer
- 類似new所得的資源一律使用unique_ptr或shared_ptr
- shared_ptr: create object to be shared or returned outside the scope
- unique_ptr: create object to be non-copy or in the local scope or in a class scope
- 從別人來的指標用weak_ptr,不用負責生成跟消滅,注意在multithread是不安全的,必須轉成shared_ptr才能使用
不建議使用tab字元做縮排,不同編輯程式對於tab字元寬度定義不同,請用空白字元做縮排。最好的方法是使編輯器支援tab鍵為4個空白。
- see Google C++ Style Guide
- see Mozilla Coding Style Guide
- see C++ Programming/Code/Style Conventions
- see Coding Style – 程式設計風格對軟體開發的影響
- see C++ Programming Style Guidelines
https://users.ece.cmu.edu/~eno/coding/CppCodingStandard.html