Вызов объявленного псевдонимом конструктора базового класса в другом пространстве имен
Я пытаюсь понять некоторые детали объявленных псевдонимами классов через C++11 using
и как / почему это влияет на вызов конструктора базового класса.
Пример кода
#include <iostream>
namespace N {
template<typename T>
struct Foo
{
// Foo(){}; // NOTE: If this line is present, error 2 goes away.
Foo(T t):value(t){};
~Foo() = default;
T value;
};
struct BarInt : public Foo<int>
{
BarInt(int t):Foo< int >(t){};
~BarInt() = default;
};
using BarFloat = Foo<float>;
};
struct A : public N::BarInt
{
A(int i=42):BarInt(i){}; //BarInt(i) or N::BarInt(i) doesn't matter here
~A() = default;
};
struct B : public N::BarFloat
{
B(float f=23.0):BarFloat(f){}; //two errors with gcc4.7.2 (with -std=gnu++11)
// B(float f=23.1):N::BarFloat(f){}; //this line seems to work.
~B() = default;
};
int main(int argc, char **argv)
{
A a;
B b;
std::cout << "a's value is "<< a.value << "\n"
<< "b's value is "<< b.value << std::endl;
return 0;
}
gcc 4.7.2 (компиляция с -std=gnu++11) генерирует две ошибки для этого кода, которые я считаю связанными (хотя я не понимаю, как...)
Ошибка 1
main.cpp: In constructor ‘B::B(float)’:
main.cpp:32:19: error: class ‘B’ does not have any field named ‘BarFloat’
Мои поиски по stackru вызвали Требуется ли пространство имен при обращении к базовому классу, в котором имя введенного класса упоминается как отправная точка для дальнейшего поиска. Однако из того, что я собрал, это объясняет, почему я могу написать конструктор для A
как я это сделал (т.е. как A(int i=42):BarInt(i){};
) и почему BarInt(i)
не должен быть квалифицирован с пространством имен N
,
Так почему же это не работает с B
? Какова разница между typedef и using в C++11?, using
такой же, как старый добрый typedef
, так что я думаю, мой вопрос для первой ошибки, как класс, объявленный псевдонимом (BarFloat
в моем примере) отличаются от обычных занятий (BarInt
в моем примере) в контексте введения имен классов. Любые указатели очень ценятся:)
Ошибка 2
main.cpp:32:29: error: no matching function for call to ‘N::Foo<double>::Foo()’
main.cpp:32:29: note: candidates are:
main.cpp:9:5: note: N::Foo<T>::Foo(T) [with T = double]
main.cpp:9:5: note: candidate expects 1 argument, 0 provided
main.cpp:6:10: note: constexpr N::Foo<double>::Foo(const N::Foo<double>&)
main.cpp:6:10: note: candidate expects 1 argument, 0 provided
Эта ошибка исчезнет, если я, как уже отмечалось в приведенном выше примере кода, введу пустой Foo()
конструктор. У меня, однако, вопрос, почему BarFloat(f)
запускает вызов в пустую Foo()
конструктор и, в этом случае, как BarFloat.value
возможно будет установлен на 23,0.
Пост скриптум
Поскольку это мой первый пост здесь: Здравствуйте, stackru и спасибо всем за огромную помощь, которую вы уже оказали мне, помогая другим в решении их проблем!
1 ответ
Когда вы наследуете от класса, производный класс может обращаться к именам базового класса (включая само имя базового класса) без квалификации. Вы эффективно наследуете имена в базовом классе. Вот почему наследование от N::BarInt
позволяет ссылаться на BarInt
в A
без квалификации.
За B
вы наследуете от Foo<double>
используя псевдоним BarFloat, но Foo<double>
не содержит BarFloat
так что вы не наследуете это имя.
Вторая ошибка только из-за первого сбоя. Поскольку компилятор не видел допустимой инициализации базы, это похоже на то, что вы вообще не инициализировали базу явно, поэтому он вынужден инициализировать ее конструктором по умолчанию, которого у вас нет.