Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Member initialization while using delegated constructor

I've started trying out the C++11 standard and i found this question which describes how to call your ctor from another ctor in the same class to avoid having a init method or the like. Now i'm trying the same thing with code that looks like this:

hpp:

class Tokenizer { public:   Tokenizer();   Tokenizer(std::stringstream *lines);   virtual ~Tokenizer() {}; private:   std::stringstream *lines; }; 

cpp:

Tokenizer::Tokenizer()   : expected('=') { }  Tokenizer::Tokenizer(std::stringstream *lines)   : Tokenizer(),     lines(lines) { } 

But this is giving me the error: In constructor ‘config::Tokenizer::Tokenizer(std::stringstream*)’: /path/Tokenizer.cpp:14:20: error: mem-initializer for ‘config::Tokenizer::lines’ follows constructor delegation I've tried moving the Tokenizer() part first and last in the list but that didn't help.

What's the reason behind this and how should i fix it? I've tried moving the lines(lines) to the body with this->lines = lines; instead and it works fine. But i would really like to be able to use the initializer list.

like image 732
lfxgroove Avatar asked Aug 30 '12 04:08

lfxgroove


People also ask

How will you initialize data member using constructor?

Initializer List is used in initializing the data members of a class. The list of members to be initialized is indicated with constructor as a comma-separated list followed by a colon. Following is an example that uses the initializer list to initialize x and y of Point class.

What is delegate constructor?

Delegating constructors can call the target constructor to do the initialization. A delegating constructor can also be used as the target constructor of one or more delegating constructors. You can use this feature to make programs more readable and maintainable.

How do you initialize a constructor member variable in a class?

Use member initializer lists to initialize your class member variables instead of assignment. A member initialization list can also be used to initialize members that are classes. When variable b is constructed, the B(int) constructor is called with value 5.

Should my constructors use initialization lists or assignment?

Initialization lists. In fact, constructors should initialize as a rule all member objects in the initialization list.


1 Answers

When you delegate the member initialization to another constructor, there is an assumption that the other constructor initializes the object completely, including all members (i.e. including the lines member in your example). You can't therefore initialize any of the members again.

The relevant quote from the Standard is (emphasis mine):

(§12.6.2/6) A mem-initializer-list can delegate to another constructor of the constructor’s class using any class-or-decltype that denotes the constructor’s class itself. If a mem-initializer-id designates the constructor’s class, it shall be the only mem-initializer; the constructor is a delegating constructor, and the constructor selected by the is the target constructor. [...]

You can work-around this by defining the version of the constructor that takes arguments first:

Tokenizer::Tokenizer(std::stringstream *lines)   : lines(lines) { } 

and then define the default constructor using delegation:

Tokenizer::Tokenizer()   : Tokenizer(nullptr) { } 

As a general rule, you should fully specify that version of the constructor that takes the largest number of arguments, and then delegate from the other versions (using the desired default values as arguments in the delegation).

like image 104
jogojapan Avatar answered Sep 21 '22 04:09

jogojapan