Types:
- feat: The new feature you're adding to a particular application
- fix: To fix a bug
- style: Feature and updates related to styling
- refactor: Refactoring a specific section of the codebase
- test: Everything related to testing
- docs: Everything related to documentation
- chore: Regular code maintanance
A scope is a phrase descibing parts of the code affected by the changes. For example "(userservice)"
Body (optional) can provide additional contextual information. For breaking changes the body MUST start with "BREAKING CHANGE".
Footer (optional) is used to reference issues effected bt the code changes. For example "Fixes #13". Can also be used to indicate breaking changes by starting with "BREAKING CHANGE".
<type>(scope): <description>
[optional body]
[optional footer]
feat(order): add purchase order button
docs(readme): document coding conventions
Avoid Hun-garian notation. This means, we never want to use variable names that contain the type.
Entities | Naming | Preview |
---|---|---|
NameSpaces | UpperCamelCase | namespace ExNamespace |
Classes and Structs | UpperCamelCase | Class ExClass |
Enums | UpperCamelCase | Enum ExEnum |
Enum Members | ALL_UPPER | ExNum::NONE ; ExNum::FIRST_HALF ; ... |
Global Variables | gUpperCamelCase | bool gRunning |
Macros | ALL_UPPER | #define USE_OPENGL |
Template Parameter | UpperCamelCase | Template<class T > |
Parameters | lowerCamelCase / pUpperCamelCase | ExMethod(int number , int* pNumber ) |
Local Variables | lowerCamelCase / pUpperCamelCase | {int number{} ;int* pNumber { new int{}}} |
Member Variables | m_ / m_p / m_v + UpperCamelCase | int m_Number ;int* m_pNumber ;std::vector m_vNumbers ; |
Member Methods | UpperCamelCase | void ExMethod (int param1, int param2); |
prefer Tabs for indents
With if
statements and for
loops we drop the brackets if it only has to execute 1 line of code or if it calls a method
if (true)
ExecuteMethod();
int num;
if (true)
num += 5;
for (;;)
ExecuteMethod();
for (;;)
num += 5;
Example for namespaces, classes, etc...
namespace ExNamespace
{
class ExClass
{
public:
ExClass()
{
}
ExMethod();
GetNum() const
{
return m_Num;
}
private:
int m_Num;
}
}
ExNamespace::ExClass::ExMethod()
{
auto lambda = []()
{
int x = 0;
return x;
};
switch (expression)
{
case 0:
{
// do stuff
break;
}
case 1:
{
// do stuff
break;
}
}
do
{
// do stuff
} while (true);
}
Prefer default initialized variables that will be set in an if
Bad example:
int value;
if(true)
{
value = 1;
}
else
{
value = 2
}
Instead we prefer this:
int value{2};
if(true)
{
value = 1;
}
Prefer using guarding if
statements for readability
Bad Example:
void Method()
{
bool val1 = true;
bool val2 = false;
bool val3 = true;
if(val1)
{
if(val2)
{
}
else
{
if(val3)
{
// code part
// where tf are we at
}
}
}
}
Guarded if Example:
void Method()
{
bool val1 = true;
bool val2 = false;
bool val3 = true;
if(!val1)
return;
if (val2)
return;
if (!val3)
return;
// here comes the code part
}
- Seperate every variable to a single line
- Initialize all member variables
- Use the same order of initialization as in the header file
- Order data types from big to low
// Header File
class ExClass
{
public:
ExClass();
private:
float m_Length;
int m_Number;
bool m_Enabled;
}
// Cpp File
ExClass::ExClass()
: m_Length{ 1.f }
, m_Number{ 5 }
, m_Enabled { true }
{
}
- Not for numeric types:
int, bool, char, float, ...
- When methods return a value
- Only when r-value type is evident
- Prefer using
auto&
to avoid copies
// 1.
int a{ 0 };
float b{ 0.5f };
// 2.
const auto& data = exClass->GetData();
// 3.
auto* pTimer = new TimerClass();
- Never use
using namespace std
! - Never use
using namespace
in a header file. This can mess with code that includes the header file wich containsusing namespace
. - Never use
using namespace
before any#include
s. This can mess with the meaning of code in someone else's header. - Use
using
instead of typedef! Modernized c++
// bad example
typedef int my_int;
// good example
using my_float = float;
- Prefer specified using over namespace using.
// bad example
using namepace std::chrono;
// good examples
using std::chrono::high_resolution_clock;
using std::chrono::duration;
Nested Namespaces
Example:
// bad example
namespace A
{
namespace B {}
}
// good example
namespace A::B {}
- Use
virtual
in Base classes - Use
virtual
andOverride
in Derived classes - Use
virtual
and= 0
for an abstract(instantiable) class - Always make destructors
virtual