ES: Expressions and statements
Expressions and statements are the lowest and most direct way of expressing actions and computation. Declarations in local scopes are statements.
For naming, commenting, and indentation rules, see NL: Naming and layout.
General rules:
- ES.1: Prefer the standard library to other libraries and to "handcrafted code"
- ES.2: Prefer suitable abstractions to direct use of language features
- ES.3: Don't repeat yourself, avoid redundant code
Declaration rules:
- ES.5: Keep scopes small
- ES.6: Declare names in for-statement initializers and conditions to limit scope
- ES.7: Keep common and local names short, and keep uncommon and non-local names longer
- ES.8: Avoid similar-looking names
- ES.9: Avoid
ALL_CAPS
names - ES.10: Declare one name (only) per declaration
- ES.11: Use
auto
to avoid redundant repetition of type names - ES.12: Do not reuse names in nested scopes
- ES.20: Always initialize an object
- ES.21: Don't introduce a variable (or constant) before you need to use it
- ES.22: Don't declare a variable until you have a value to initialize it with
- ES.23: Prefer the
{}
-initializer syntax - ES.24: Use a
unique_ptr<T>
to hold pointers - ES.25: Declare an object
const
orconstexpr
unless you want to modify its value later on - ES.26: Don't use a variable for two unrelated purposes
- ES.27: Use
std::array
orstack_array
for arrays on the stack - ES.28: Use lambdas for complex initialization, especially of
const
variables - ES.30: Don't use macros for program text manipulation
- ES.31: Don't use macros for constants or "functions"
- ES.32: Use
ALL_CAPS
for all macro names - ES.33: If you must use macros, give them unique names
- ES.34: Don't define a (C-style) variadic function
Expression rules:
- ES.40: Avoid complicated expressions
- ES.41: If in doubt about operator precedence, parenthesize
- ES.42: Keep use of pointers simple and straightforward
- ES.43: Avoid expressions with undefined order of evaluation
- ES.44: Don't depend on order of evaluation of function arguments
- ES.45: Avoid "magic constants"; use symbolic constants
- ES.46: Avoid narrowing conversions
- ES.47: Use
nullptr
rather than0
orNULL
- ES.48: Avoid casts
- ES.49: If you must use a cast, use a named cast
- ES.50: Don't cast away
const
- ES.55: Avoid the need for range checking
- ES.56: Write
std::move()
only when you need to explicitly move an object to another scope - ES.60: Avoid
new
anddelete
outside resource management functions - ES.61: Delete arrays using
delete[]
and non-arrays usingdelete
- ES.62: Don't compare pointers into different arrays
- ES.63: Don't slice
- ES.64: Use the
T{e}
notation for construction - ES.65: Don't dereference an invalid pointer
Statement rules:
- ES.70: Prefer a
switch
-statement to anif
-statement when there is a choice - ES.71: Prefer a range-
for
-statement to afor
-statement when there is a choice - ES.72: Prefer a
for
-statement to awhile
-statement when there is an obvious loop variable - ES.73: Prefer a
while
-statement to afor
-statement when there is no obvious loop variable - ES.74: Prefer to declare a loop variable in the initializer part of a
for
-statement - ES.75: Avoid
do
-statements - ES.76: Avoid
goto
- ES.77: Minimize the use of
break
andcontinue
in loops - ES.78: Don't rely on implicit fallthrough in
switch
statements - ES.79: Use
default
to handle common cases (only) - ES.84: Don't try to declare a local variable with no name
- ES.85: Make empty statements visible
- ES.86: Avoid modifying loop control variables inside the body of raw for-loops
- ES.87: Don't add redundant
==
or!=
to conditions
Arithmetic rules:
- ES.100: Don't mix signed and unsigned arithmetic
- ES.101: Use unsigned types for bit manipulation
- ES.102: Use signed types for arithmetic
- ES.103: Don't overflow
- ES.104: Don't underflow
- ES.105: Don't divide by integer zero
- ES.106: Don't try to avoid negative values by using
unsigned
- ES.107: Don't use
unsigned
for subscripts, prefergsl::index
ES.1: Prefer the standard library to other libraries and to "handcrafted code"
Reason
Code using a library can be much easier to write than code working directly with language features, much shorter, tend to be of a higher level of abstraction, and the library code is presumably already tested. The ISO C++ Standard Library is among the most widely known and best tested libraries. It is available as part of all C++ implementations.
Example
auto sum = accumulate(begin(a), end(a), 0.0); // good
a range version of accumulate
would be even better:
auto sum = accumulate(v, 0.0); // better
but don't hand-code a well-known algorithm:
int max = v.size(); // bad: verbose, purpose unstated
double sum = 0.0;
for (int i = 0; i < max; ++i)
sum = sum + v[i];
Exception
Large parts of the standard library rely on dynamic allocation (free store). These parts, notably the containers but not the algorithms, are unsuitable for some hard-real-time and embedded applications. In such cases, consider providing/using similar facilities, e.g., a standard-library-style container implemented using a pool allocator.
Enforcement
Not easy. ??? Look for messy loops, nested loops, long functions, absence of function calls, lack of use of built-in types. Cyclomatic complexity?
ES.2: Prefer suitable abstractions to direct use of language features
Reason
A "suitable abstraction" (e.g., library or class) is closer to the application concepts than the bare language, leads to shorter and clearer code, and is likely to be better tested.
Example
vector<string> read1(istream& is) // good
{
vector<string> res;
for (string s; is >> s;)
res.push_back(s);
return res;
}
The more traditional and lower-level near-equivalent is longer, messier, harder to get right, and most likely slower:
char** read2(istream& is, int maxelem, int maxstring, int* nread) // bad: verbose and incomplete
{
auto res = new char*[maxelem];
int elemcount = 0;
while (is && elemcount < maxelem) {
auto s = new char[maxstring];
is.read(s, maxstring);
res[elemcount++] = s;
}
*nread = elemcount;
return res;
}
Once the checking for overflow and error handling has been added that code gets quite messy, and there is the problem remembering to delete
the returned pointer and the C-style strings that array contains.
Enforcement
Not easy. ??? Look for messy loops, nested loops, long functions, absence of function calls, lack of use of built-in types. Cyclomatic complexity?
ES.3: Don't repeat yourself, avoid redundant code
Duplicated or otherwise redundant code obscures intent, makes it harder to understand the logic, and makes maintenance harder, among other problems. It often arises from cut-and-paste programming.
Use standard algorithms where appropriate, instead of writing some own implementation.
Example
void func(bool flag) // Bad, duplicated code.
{
if (flag) {
x();
y();
}
else {
x();
z();
}
}
void func(bool flag) // Better, no duplicated code.
{
x();
if (flag)
y();
else
z();
}
Enforcement
- Use a static analyzer. It will catch at least some redundant constructs.
- Code review