Наследование конструкторов
Почему этот код:
class A
{
public:
explicit A(int x) {}
};
class B: public A
{
};
int main(void)
{
B *b = new B(5);
delete b;
}
Результат этих ошибок:
main.cpp: In function ‘int main()’:
main.cpp:13: error: no matching function for call to ‘B::B(int)’
main.cpp:8: note: candidates are: B::B()
main.cpp:8: note: B::B(const B&)
Не следует ли B наследовать конструктор A?
(используется gcc)
Ответы
Ответ 1
Если ваш компилятор поддерживает стандарт С++ 11, существует наследование конструктора с using
(предназначен для каламбура). Для получения дополнительной информации см. Статью Wikipedia С++ 11. Ты пишешь:
class A
{
public:
explicit A(int x) {}
};
class B: public A
{
using A::A;
};
Это все или ничего - вы не можете наследовать только некоторые конструкторы, если вы пишете это, вы наследуете их все. Чтобы наследовать только выбранные, вам нужно написать отдельные конструкторы вручную и вызвать из них базовый конструктор.
Исторически конструкторы не могли быть унаследованы в стандарте С++ 03. Вам нужно было наследовать их вручную по очереди, вызывая базовую реализацию самостоятельно.
Ответ 2
Конструкторы не наследуются. Они называются неявным или явным образом дочерним конструктором.
Компилятор создает конструктор по умолчанию (один без аргументов) и конструктор копии по умолчанию (один с аргументом, который является ссылкой на тот же тип). Но если вы хотите, чтобы конструктор принял int, вы должны явно определить его.
class A
{
public:
explicit A(int x) {}
};
class B: public A
{
public:
explicit B(int x) : A(x) { }
};
UPDATE. В С++ 11 конструкторы могут быть наследованы. См. Ответ Suma для деталей.
Ответ 3
Вы должны явно определить конструктор в B и явно вызвать конструктор для родителя.
B(int x) : A(x) { }
или
B() : A(5) { }
Ответ 4
Это прямо из Страница Bjarne Stroustrup:
Если вы так решите, вы все равно можете стрелять в ногу, наследуя конструкторы в производном классе, в котором вы определяете новые переменные-члены, требующие инициализации:
struct B1 {
B1(int) { }
};
struct D1 : B1 {
using B1::B1; // implicitly declares D1(int)
int x;
};
void test()
{
D1 d(6); // Oops: d.x is not initialized
D1 e; // error: D1 has no default constructor
}
Ответ 5
Как использовать функцию шаблона для привязки всех конструкторов?
template <class... T> Derived(T... t) : Base(t...) {}
Ответ 6
Правильный код
class A
{
public:
explicit A(int x) {}
};
class B: public A
{
public:
B(int a):A(a){
}
};
main()
{
B *b = new B(5);
delete b;
}
Ошибка - это b/c. Класс B не имеет конструктора параметров, а во-вторых, он должен иметь инициализатор базового класса для вызова конструктора конструктора параметров базового класса