F: Functions
A function specifies an action or a computation that takes the system from one consistent state to the next. It is the fundamental building block of programs.
It should be possible to name a function meaningfully, to specify the requirements of its argument, and clearly state the relationship between the arguments and the result. An implementation is not a specification. Try to think about what a function does as well as about how it does it. Functions are the most critical part in most interfaces, so see the interface rules.
Function rule summary:
Function definition rules:
- F.1: "Package" meaningful operations as carefully named functions
- F.2: A function should perform a single logical operation
- F.3: Keep functions short and simple
- F.4: If a function might have to be evaluated at compile time, declare it
constexpr
- F.5: If a function is very small and time-critical, declare it inline
- F.6: If your function must not throw, declare it
noexcept
- F.7: For general use, take
T*
orT&
arguments rather than smart pointers - F.8: Prefer pure functions
- F.9: Unused parameters should be unnamed
- F.10: If an operation can be reused, give it a name
- F.11: Use an unnamed lambda if you need a simple function object in one place only
Parameter passing expression rules:
- F.15: Prefer simple and conventional ways of passing information
- F.16: For "in" parameters, pass cheaply-copied types by value and others by reference to
const
- F.17: For "in-out" parameters, pass by reference to non-
const
- F.18: For "will-move-from" parameters, pass by
X&&
andstd::move
the parameter - F.19: For "forward" parameters, pass by
TP&&
and onlystd::forward
the parameter - F.20: For "out" output values, prefer return values to output parameters
- F.21: To return multiple "out" values, prefer returning a struct or tuple
- F.60: Prefer
T*
overT&
when "no argument" is a valid option
Parameter passing semantic rules:
- F.22: Use
T*
orowner<T*>
to designate a single object - F.23: Use a
not_null<T>
to indicate that "null" is not a valid value - F.24: Use a
span<T>
or aspan_p<T>
to designate a half-open sequence - F.25: Use a
zstring
or anot_null<zstring>
to designate a C-style string - F.26: Use a
unique_ptr<T>
to transfer ownership where a pointer is needed - F.27: Use a
shared_ptr<T>
to share ownership
- F.42: Return a
T*
to indicate a position (only) - F.43: Never (directly or indirectly) return a pointer or a reference to a local object
- F.44: Return a
T&
when copy is undesirable and "returning no object" isn't needed - F.45: Don't return a
T&&
- F.46:
int
is the return type formain()
- F.47: Return
T&
from assignment operators - F.48: Don't return
std::move(local)
- F.49: Don't return
const T
Other function rules:
- F.50: Use a lambda when a function won't do (to capture local variables, or to write a local function)
- F.51: Where there is a choice, prefer default arguments over overloading
- F.52: Prefer capturing by reference in lambdas that will be used locally, including passed to algorithms
- F.53: Avoid capturing by reference in lambdas that will be used non-locally, including returned, stored on the heap, or passed to another thread
- F.54: When writing a lambda that captures
this
or any class data member, don't use[=]
default capture - F.55: Don't use
va_arg
arguments - F.56: Avoid unnecessary condition nesting
Functions have strong similarities to lambdas and function objects.
See also: C.lambdas: Function objects and lambdas