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.
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.
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.
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.
Initialization lists. In fact, constructors should initialize as a rule all member objects in the initialization list.
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).
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With