Constant Expressions: constexpr
In this lesson, we will see how constexpr is used in modern C++.
Constant expressions #
With constexpr
, we can define an expression that can be evaluated at compile time. constexpr
can be used for variables, functions, and user-defined types. An expression that is evaluated at compile time has a lot of advantages.
A constant expression:
- can be evaluated at compile time
- gives the compiler deep insight into the code
- is implicitly thread-safe
- can be constructed in the read-only memory (ROM-able)
constexpr - variables and objects #
If we declare a variable as constexpr
, the compiler will evaluate it at compile time. This holds true not only for built-in types but also for instantiations of user-defined types. There are a few serious restrictions for objects in order to evaluate them at compile time.
To make our lives easier, we will use types built into the C++ library like bool
, char
, int
, and double
. We will call the remaining data types user-defined data types. These are all std::string
types. User-defined types typically hold built-in types.
Variables #
By using the keyword, constexpr
, a variable can be made a constant expression.
constexpr double myDouble= 5.2;
Therefore, we can use the variable in contexts that require a constant expression, e.g., defining the size of an array. This has to be done at compile time.
For the declaration of a constexpr
variable, we have to keep a few rules in mind.
The variable:
- is implicitly
const
. - has to be initialized.
- requires a constant expression for initialization.
The rules make sense. If we evaluate a variable at compile-time, the variable can only depend on values that can be evaluated at compile time.
User-Defined types #
Objects are created by the invocation of the constructor. The constructor has a few special rules.
A constexpr
constructor:
-
can only be invoked with constant expressions.
-
cannot use exception handling.
-
has to be declared as
default
ordelete
or the function body must be empty (C++11).
A constexpr
user-defined type cannot have virtual base classes. It requires that each base object and each non-static member is initialized in the initialization list of the constructor, or directly in the class body. Consequently, it holds that each used constructor (e.g of a base class) has to be a constexpr
constructor and that the applied initializers have to be constant expressions.
Example #
struct MyDouble{
double myVal;
constexpr MyDouble(double v): myVal(v){}
constexpr double getVal(){return myVal;}
};
-
The constructor has to be both empty and a constant expression.
-
The user-defined type can have methods that may or may not be constant expressions.
-
Instances of
MyDouble
can be instantiated at compile time.
Functions #
constexpr
functions are functions that have the potential to run at compile time. This means we can perform a lot of calculations at compile time, with the results available at runtime and stored as a constant in the ROM. In addition, constexpr
functions are implicitly inline
.
A constexpr
function can be invoked with a non-constexpr
value. In this case, the function runs at runtime. A constexpr
function is executed at compile-time when it is used in an expression that is evaluated at compile time (such as static_assert
or the definition of a C-array) or when the result is requested at compile time:
constexpr auto res = constexprFunction()
For constexpr
functions, there are a few restrictions:
The function:
- has to be non-virtual.
- has to have arguments and a
return
value of a literal type.constexpr
variables are of literal types. - can only have one return statement.
- must return a value.
- will be executed at compile time if invoked within a constant expression.
- can only have a function body consisting of a return statement.
- must have a constant return value.
- is implicitly
inline
.
Example #
constexpr int fac(int n)
{return n > 0 ? n * fac(n-1): 1;}
constexpr int gcd(int a, int b){
return (b== 0) ? a : gcd(b, a % b);
Functions with C++14 #
The syntax of constexpr
functions was massively improved with the change from C++11 to C++14. In C++11, we had to remember which features we were able to use in a constexpr
function. With C++14, we only have to remember which features we can’t use in a constexpr
function.
constexpr
functions in C++14
- can have variables that have to be initialized by a constant expression.
- can have loops.
- cannot have
static
orthread_local
data. - can have conditional jump instructions or loop instructions.
- can have more than one instruction.
Example: Magic of constexpr
functions #
Get hands-on with 1300+ tech skills courses.