C++: почему конструктор моего DerivedClass не имеет доступа к защищенному полю BaseClass?
У меня есть конструктор, пытающийся инициализировать поле в базовом классе. Компилятор жалуется. Поле защищено, поэтому производные классы должны иметь доступ.
//The base class:
class BaseClass
{
public:
BaseClass(std::string);
BaseClass(const BaseClass& orig);
virtual ~BaseClass();
const std::string GetData() const;
void SetData(const std::string& data);
protected:
BaseClass();
std::string m_data;
};
BaseClass::BaseClass(const std::string data) : m_data(data) { }
BaseClass::BaseClass() { }
BaseClass::BaseClass(const BaseClass& orig) { }
BaseClass::~BaseClass() { }
void BaseClass::SetData(const std::string& data)
{
m_data = data;
}
const std::string BaseClass::GetData() const
{
return m_data;
}
//The derived class:
class DerivedClass : public BaseClass
{
public:
DerivedClass(std::string data);
DerivedClass(const DerivedClass& orig);
virtual ~DerivedClass();
private:
};
DerivedClass::DerivedClass(std::string data) : m_data(data) { } //ERROR HERE
DerivedClass::DerivedClass(const DerivedClass& orig) { }
DerivedClass::~DerivedClass() { }
//ошибка компилятора
DerivedClass.cpp: 3: error: class 'DerivedClass’ не имеет поля с именем 'm_data'
любая помощь очень ценится. Заранее спасибо.
6 ответов
вы не можете инициализировать m_data в конструкторе производного класса, но вместо этого передаете его в качестве аргумента конструктору базового класса.
что есть: DerivedClass::DerivedClass(std::string data) : BaseClass(data) { }
в списке инициализаторов можно просто задать значения атрибутов одного класса. Для доступа к нему необходимо присвоить значение в теле конструктора:
DerivedClass:: DerivedClass (std::string data) {m_data = data;}
или, если копировать объект дорого, вы передаете m_data в качестве аргумента конструктору родительского класса:
DerivedClass:: DerivedClass(std:: string data) : BaseClass(data) {}
Совет: передайте свои данные в качестве ссылки, чтобы предотвратить конструктор копирования.
Подробнее см. здесь:порядок инициализации конструкторов C++.
вы не "обращаетесь" к m_data - вы инициализируете его. Однако он уже был инициализирован в ctor базового класса. Если вы хотите изменить его значение, назначьте ему в теле вашего ctor:
DerivedClass::DerivedClass(std::string data)
{
m_data = data;
}
вам нужно вызвать конструктор базового класса следующим образом:
DerivedClass::DerivedClass(std::string data) : BaseClass(data) {
}
каждый класс должен отвечать за инициализацию его членов.
списки инициализаторов могут использоваться только для инициализации полей, принадлежащих данному типу. Инициализация полей базового класса в списках инициализаторов не является законной, поэтому вы получаете эту ошибку. В противном случае поле доступно в DerivedClass
у вашего конструктора производного класса есть доступ, но вы не можете назначить ему в списке инициализации. Измените конструктор на:
DerivedClass::DerivedClass(const std::string& data)
: BaseClass(data)
{
}
альтернативно, если нет подходящего конструктора базового класса, и вы не можете добавить его, вы можете изменить конструктор на:
DerivedClass::DerivedClass(const std::string& data)
{
m_data = data;
}