Tuesday, January 18, 2011

Constructor initialization lists

Up until now, we’ve been initializing our class member data in the constructor using the assignment operator. For example:
01class Something
02{
03private:
04    int m_nValue;
05    double m_dValue;
06    int *m_pnValue;
07
08public:
09    Something()
10    {
11        m_nValue = 0;
12        m_dValue = 0.0;
13        m_pnValue = 0;
14    }
15};
When the class’s constructor is executed, m_nValue, m_dValue, and m_chValue are created. Then the body of the constructor is run, where the member data variables are assigned values. This is similar to the flow of the following code in non-object-oriented C++:
1int nValue;
2double dValue;
3int *pnValue;
4
5nValue = 0;
6dValue = 0.0;
7pnValue = 0;
While this does not exhibit good style, it is valid within the syntax of the C++ language.
So far, the classes that we have written have only included non-const or pointer member variables. However, what happens when we want to use const or reference variables as member variables? As you have learned in previous lessons, const and reference variables must be initialized on the line they are declared. Consider the following example:
01class Something
02{
03private:
04    const int m_nValue;
05public:
06    Something()
07    {
08        m_nValue = 5;
09    }
10};
This produces code similar to the following:
1const int nValue; // error, const vars must be assigned values immediately
2nValue = 5;
Consequently, assigning const or reference member variables values in the body of the constructor is not sufficient.
Initialization lists
C++ provides another way of initializing member variables that allows us to initialize member variables when they are created rather than afterwards. This is done through use of an initialization list.
In the lesson on basic addressing and variable declaration, you learned that you could assign values to variables in two ways: explicitly and implicitly:
1int nValue = 5; // explicit assignment
2double dValue(4.7); // implicit assignment
Using an initialization list is very similar to doing implicit assignments.
Let’s take a look at our top example again. Here’s the code that does explicit assignments in the constructor body:
01class Something
02{
03private:
04    int m_nValue;
05    double m_dValue;
06    int *m_pnValue;
07
08public:
09    Something()
10    {
11        m_nValue = 0;
12        m_dValue = 0.0;
13        m_pnValue = 0;
14    }
15};
Now let’s write the same code using an initializer list:
01class Something
02{
03private:
04    int m_nValue;
05    double m_dValue;
06    int *m_pnValue;
07
08public:
09    Something() : m_nValue(0), m_dValue(0.0), m_pnValue(0)
10    {
11    }
12};
The initialization list is inserted after the constructor parameters, begins with a colon (:), and then lists each variable to initialize along with the value for that variable separated by a comma. Note that we no longer need to do the explicit assignments in the constructor body, since the initializer list replaces that functionality. Also note that the initialization list does not end in a semicolon.
Here’s an example of a class that has a const member variable:
1class Something
2{
3private:
4    const int m_nValue;
5public:
6    Something(): m_nValue(5)
7    {
8    }
9};
We strongly encourage you to begin using this new syntax (even if you aren’t using const or reference member variables) as initialization lists are required when doing composition and inheritance (subjects we will be covering shortly).

No comments: