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;
}