严谨编码风格的重要性:
1.便于代码合并,满足团队开发的需要
2.合格程序员的必备素质
3.防止编码错误
4.易读易理解
Google C++编码规范(中文版)
https://zh-google-styleguide.readthedocs.io/en/latest/google-cpp-styleguide/
1.头文件尽可能多的使用前向声明
好处:较少的文件依赖,减少编译时间. 更加明确的类的依赖关系.
1 //.h中类B使用类A,使用前向声明就不需要包含类A的头文件,减少编译工作量 2 class CA; //前向声明一个类A 3 class CB 4 { 5 public: 6 CA* m_pA;//声明类A的指针 7 CA* GetClassA(CA *pA, CA &a); //声明使用类A作为形参或返回值类型的函数 8 //CA m_A; Error,不能定义类A的对象 9 };
2.函数的参数顺序,输入参数在前,输出参数在后
输入参数一般为传值或者指针,引用,前面加const
输出参数一般为非常量指针或者引用
好处:根据参数的位置明确参数的作用,一目了然
1 //输入参数: nInput cInput szInput 2 //输出参数: nOutPut 3 void Fun(const int nInput, const char cInput, const char* pStrInput, int &nOutPut);
3.头文件的包含顺序
stdafx.h预编译头文件
先包含当前源文件对应的头文件
C系统头文件
C++系统头文件
其它库头文件
本项目其它头文件
每个区域之间空一行
好处:增加可读性
1 #include "stdafx.h" //预编译头文件 2 3 #include "test.h" //当前源代码对应的头文件 4 5 #include <stdio.h> //C标准库 6 7 #include <iostream> //C++标准库 8 9 #include "library/read.h" //其它库头文件 10 #include "library/write.h" 11 12 #include "one.h" //当前项目下其它头文件 13 #include "two.h"
4.局部变量初始化,声明变量后立即进行赋值初始化
好处:避免未初始化产生的潜在错误
1 int nA = 0; 2 char szB = '\0'; 3 float fC = 0.00f; 4 char szBuff[16] = {0}; //注意:初始化时指定的的元素个数比数组大小少,剩下的元素都回被初始化为0 5 StPeople tPeople = {0};
5.类的初始化
1 class CStudent 2 { 3 public: 4 CStudent():m_nAge(0),m_fScore(0.00f){} 5 CStudent(int nAge, float fScore):m_nAge(nAge),m_fScore(fScore){}; 6 private: 7 int m_nAge; 8 float m_fScore; 9 }; 10 int _tmain(int argc, _TCHAR* argv[]) 11 { 12 CStudent student; //使用无参的构造函数初始化 m_nAge=0 m_fScore=0.00f 13 CStudent studentTwo(20,99.5); //使用初始化例表构造函数初始化 m_nAge=20 m_fScore=99.5f 14 }
6.局部变量是个对象,应该避免多次调用构造与析构
1 //这里循环一千次,每次循环都得实例化对象,调用构造与析构
2 for(int i = 0; i<1000; i++)
3 {
4 CTest test;
5 test.Out();
6 }
7 //下面同样循环一千次,在外部实例化对象,只调用一次构造与析构
8 CTest testTwo;
9 for(int j = 0; j<1000; j++)
10 {
11 testTwo.Out();
12 }
7.结构体和类使用, 只有当成员全部为数据时使用结构体,其它时一律使用类.
C++中结构体和类起到同样作用,唯一区别是结构体默认成员和方法是public,而类默认是private
8.操作符重载,尽量不要用操作符重载,定义一个函数来处理显得更加直观
1 class CPerson 2 { 3 public: 4 CPerson(int nAge):m_nAge(nAge){} 5 //重载==运算符 6 bool operator==(const CPerson &ps) 7 { 8 if (this->m_nAge == ps.m_nAge) 9 { 10 return true; 11 } 12 return false; 13 } 14 //使用Equal函数 15 bool Equal(const CPerson &ps) 16 { 17 return this->m_nAge == ps.m_nAge; 18 } 19 private: 20 int m_nAge; 21 }; 22 int _tmain(int argc, _TCHAR* argv[]) 23 { 24 CPerson person1(20); 25 CPerson person2(20); 26 if(person1 == person2) 27 { 28 printf("person1 == person2\r\n"); 29 } 30 31 if(person1.Equal(person2)) 32 { 33 printf("person1 == person2\r\n"); 34 } 35 36 getchar(); 37 }
9.将类的数据成员私有化,提供Get Set方法进行访问, 不使用对象,xxxx方式访问
好处:提高代码封装性,有利于调试.
当数据成员出错时,只需在Get Set方法下断调试即可. 反之使用对象.xxx方式访问,出错时需要在所有调用处下断,不利于调试.
1 class CPerson 2 { 3 public: 4 CPerson(int nAge):m_nAge(nAge){} 5 int GetAge() 6 { 7 return this->m_nAge; 8 } 9 void SetAge(const int nAge) 10 { 11 this->m_nAge = nAge; 12 } 13 private: 14 int m_nAge; 15 }; 16 int _tmain(int argc, _TCHAR* argv[]) 17 { 18 CPerson person1(20); 19 printf("age = %d\r\n",person1.GetAge()); 20 person1.SetAge(30); 21 getchar(); 22 }
10.类中的声明顺序, 先声明public块,再声明protected块,再声明private块,这三块中声明的顺序一致
1 class CPerson 2 { 3 public: 4 //typedef 5 typedef vector<int> VectorInt; 6 //常量 7 const int m_kNum; 8 //构造和析构函数 9 CPerson(int nAge):m_kNum(100),m_nAge(nAge){} 10 ~CPerson(){} 11 //成员函数 12 int GetAge() 13 { 14 return this->m_nAge; 15 } 16 void SetAge(const int nAge) 17 { 18 this->m_nAge = nAge; 19 } 20 //数据成员 21 protected: 22 //声明顺序同上 23 private: 24 //声明顺序同上 25 int m_nAge; 26 };
11.编写短小的函数,尽量不要超过40行
好处:方便提取重复代码进行封装,便于他人阅读和修改,方便调试
12.不使用函数重载,直接使用函数名进行区分
好处:过多的函数重载种类会使调用者无从选择
1 //函数重载 2 CPerson* GetPerson(const int nAge); 3 CPerson* GetPerson(const Sex emSex); 4 CPerson* GetPerson(const char* pStrName); 5 6 //直接利用函数名进行区分 7 CPerson* GetPersonFromAge(const int nAge); 8 CPerson* GetPersonFromSex(const Sex emSex); 9 CPerson* GetPersonFromName(const char* pStrName);
13.函数参数禁止使用缺省参数
好处:避免他人调用函数时错误的理解
14.迭代器上使用前置自增自减(++i --i)
好处:提高迭代器访问的执行效率, 相比后置自增自减而言
15.当函数参数为对象时,尽量使用指针或者引用方式传递
//直接传递,需要构造一个新的对象,这个新的对象又得构造和析构,影响性能 void GetPeoson(const CPerson person); //使用指针或引用方式传递实参,不需要构造新的对象 void GetPesSon(const CPerson *pPerson); void GetPeoson(const CPerson &person);
16.使用sizeof(变量名),而不是使用sizeof(数据类型)
好处:当修改变量的数据类型后,sizeof不需要同步更新成新的数据类型
17.格式要求
函数中返回值与函数名在同一行,如果参数过多需要进行换行并且对齐,每一行一个参数
1 void SetWindow(HWND hWnd, 2 int nWidth, 3 int nHeight, 4 int nX, 5 int nY, 6 DWORD dwStyle);
if和else if语句块的括号需要另起一行,方便阅读
1 if(aa > bb) 2 { 3 //...... 4 } 5 else if(aa > cc) 6 { 7 //..... 8 }
适当添加空格,使代码美观更利于阅读
1 int nA, nB, nC; 2 nC = nA + nB;