Initializing Variables in Embedded C++

C++ is infamous for the number of ways you can initialize a variable. If you are working with large distinct codebases (aka, you are a software engineer), you need to know them all. However, if you are an embedded C engineer looking to improve your C++ skills, you just need to know the few ways that work best for you.

Variable Initialization Options

When a local variable is declared, you have several ways of initializing it.

//default initialization
int variable;
//Copy/move assignment initialization
int variable = 10;
//direct initialization using parenthesis (aka round brackets)
int variable(10); 
//brace (aka curly brackets) initialization
int variable{10}; //direct brace
int variable = {10}; //copy brace
int variable{}; //value initialization
int variable = {}; //value initialization

//and the auto versions
auto variable = 10;
auto variable{10}; //direct brace
auto variable = int{10};
auto variable(10); //direct curly
auto variable = int(10);
auto variable = int{};
auto variable = int();

There are a few advantages of using auto. First, it is impossible to declare a variable using auto and not initialize it. Second, it allows you to avoid, sometimes expensive, implicit variable conversions. And, it has left-to-right readability to it. For example, auto variable = int{10} reads as: variable is an int with an initial value of 10.

MyClass and MyStruct

We are going to use this simple class below as we talk about each of these.


class MyClass {
public:
  MyClass() = default;
  explicit MyClass(int value) : x{value} {}
  MyClass(int x_value, int y_value) : x{x_value}, y{y_value} {}
private:
  int x{};
  int y{};
};

struct MyStruct {
  int x{};
  int y{};
};

In C++, the only difference between struct and class is that class members are private by default and struct members are public by default.

Default Initialization

For a fundamental type (like int, float, void*), default initialization on the stack leaves the value in an indeterminate state. You shouldn’t ever do this. Consider this example:

int variable;
read(fd, &variable, sizeof(int));

//this is better
//create a custom function to assign the variable
//we can use const so that we know this variable isn't
//modified later
const int variable = read_variable_from_file(fd);
//it we use auto, we avoid a potential conversion if we pick the wrong type
const auto variable = read_variable_from_file(fd);

For a class or struct, the default initialization calls the default constructor. In the case of MyClass, the default constructor uses value initialization with the empty braces which assign the value of x and y to zero. The same is true for MyStruct.

We will discuss below how to ensure all members of a class or struct are appropriately initialized during construction.

Copy Initialization

For fundamental types, the value assigned is copied to the variable. For class and struct types, the value is either copied or moved depending on the class or struct. If the assigned value is an rvalue, the value is moved. If it is an lvalue, the value is copied.

An lvalue is a value that has a location in memory. That is you can use a pointer/reference with it. An rvalue is defined as not being an lvalue. It is a temporary value that may only exist in a register.

//Move the temporary MyClass() to my_class
MyClass my_class = MyClass();
auto my_class = MyClass();
//copy my_class to my_class2
MyClass my_class2 = my_class;
auto my_class2 = my_class;

Not everything in C++ is movable. If a class cannot be moved, it is copied. The compiler will make this decision without any warnings or errors.

The compiler will also apply copy and move elision optimizations. For example, the following two lines of code will compile to the same instructions. The copy (or move) in the second line is omitted. my_copyied_class is directly constructed using MyClass(5,6) rather than using MyClass() and doing a copy (or move) from the temporary MyClass(5,6).

MyClass my_default_class(5,6);
MyClass my_copyied_class = MyClass(5,6);

Copy and move elision apply to all relevant (those that use =) initialization styles.

Direct Initialization

int variable(10);
MyClass my_class(5,10);
MyStruct my_struct(5,10); //this is a compiler error

For fundamental types, direct initialization is the same as copy initialization. For MyClass, the constructor taking two integers as arguments is used MyClass(int x_value, int y_value). The compiler searches all the constructors for the one that matches the call site. MyStruct does not provide a constructor that takes two arguments so the compilation of that line will fail.

Brace Initialization

Brace initialization is also known as uniform initialization because it can be used everywhere. Direct initialization is limited to matching constructors.

For fundamental types, brace initialization offers two advantages over other methods. Brace initialization disallows narrowing types. For example:

int w = 4.5f; //implicit conversion from float to int
int w{4.5f}; //compilation error

Brace initialization also allows easy access to value initialization.

int variable{};
int variable = {};

For fundamental types, value initialization assigns the values to zero. This also works for plain old data structures (c-style structs).

typedef struct {
  int x;
  int y;
  int z;
} c_style_t;

//this will assign all values to zero
c_style_t c_style = {};
auto c_cstyle = c_style_t{};

//WAY better than writing this
//but in many cases this is exactly what the
//compiler does
c_style_t c_style; //uninitialized
memset(&c_style, 0, sizeof(c_style));

Brace initialization on class or struct will either match the constructor or it will match the list in the {} with public members of the struct or class. If not all members are public, a constructor must match.

MyClass my_class{5,10};
auto my_class = MyClass{5,10};
auto my_class{MyClass{5,10}};
MyClass my_class = {5,10};

MyStruct my_struct{5,10};
MyStruct my_struct = {5,10};
auto my_struct = MyStruct{5,10};
auto my_struct{MyStruct{5,10}};

Another nice feature of brace initialization is you can (and should) include the member names to make the code clear. This is the same syntax as C but the members must be in the same order they are declared or the compilation will fail.

MyStruct my_struct{.x = 5, .y = 10};
auto my_struct{MyStruct{.x = 5, .y = 10}};
MyStruct my_struct = {.x = 5, .y = 10};
auto my_struct = MyStruct{.x = 5, .y = 10};
MyStruct my_struct = {.y = 5, .x = 10}; //compilation error

//this doesn't work with private members
MyClass my_class{.x = 5, .y = 10}; //compilation error

Member Variable Initialization

Member variables within classes have a few things to consider in addition to local variable initialization.

First, variables are initialized (including calling constructors) in the order they are declared. So x is assigned zero before y. The constructor of the enclosing class is called last. Order isn’t too important in MyClass but in some cases, it matters. When a class is destructed, the destructors are called in the exact opposite order as the constructors.

In MyClass and MyStruct, we used value initialization to assign the variables x and y. We could have used any of the initialization types. In MyClass, we also use a member constructor list with MyClass(int x_value, int y_value). The important thing to understand is that if a variable appears in a member constructor list, the declaration initialization will not be executed.

//x is only initialized with x_value, x{} is not executed
//y is initialized from int y{}
explicit MyClass(int value) : x{value} {
  y = 10; //both y{} and y = 10 are executed -- don't do this
  //just put , y{10} after x{value} above
}
//neither x{} nor y{} is executed
MyClass(int x_value, int y_value) : x{x_value}, y{y_value} {}

The End

There are a bunch of ways to initialize variables in C++ (and if we are being honest, it is a bit of a mess). In most cases, the differences are a matter of style. But remember, brace initialization and auto have several advantages. With brace initialization, the typing is more strict for fundamental types and can be used for both constructors and directly initializing public members. auto guarantees initialization, avoids implicit conversions, and is left-to-right readable. My preference is:

const auto value = int{10};
const auto my_class = MyClass{5,10};