Бьярн Страустрап - Справочное руководство по C++
- Название:Справочное руководство по C++
- Автор:
- Жанр:
- Издательство:неизвестно
- Год:неизвестен
- ISBN:нет данных
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Бьярн Страустрап - Справочное руководство по C++ краткое содержание
Справочное руководство по C++ - читать онлайн бесплатно полную версию (весь текст целиком)
Интервал:
Закладка:
список-базовых:
спецификация-базовых
список-базовых , спецификация-базовых
спецификация-базовых:
полное-имя-класса
virtual спецификация-доступа optполное-имя-класса
спецификация-доступа virtual optполное-имя-класса
спецификация-доступа:
private
protected
public
Конструкция имя-класса в спецификации-базовых должна обозначать ранее описанный класс (§R.9), который называется базовым по отношению к определяемому классу. Говорят, что класс является производным от своих базовых классов. Назначение конструкции спецификация-доступа объясняется в §R.11. К членам базового класса, если только они не переопределены в производном классе, можно обращаться так, как будто они являются членами производного класса. Говорят, что производный класс наследует члены базового класса. С помощью операции разрешения области видимости :: (§R.5.1) к члену базового класса можно обращаться явно. Такое обращение возможно и в том случае, когда имя члена базового класса переопределено в производном классе. Производный класс сам может выступать как базовый при контроле доступа, см. §R.11.2. Указатель на производный класс может неявно преобразовываться в указатель на однозначно определенный и доступный базовый класс (§R.4.6). Ссылка на производный класс может неявно преобразовываться в ссылку на однозначно определенный и доступный базовый класс (§R.4.7).
Рассмотрим пример:
class base {
public:
int a, b;
};
class derived: public base {
public:
int b, c;
};
void f()
{
derived d;
d.a = 1;
d.base::b = 2;
d.b = 3;
d.c = 4;
base* bp = &d; // стандартное преобразование derived* в base*
}
Здесь присваиваются значения четырем членам d, а bp настраивается на d.
Класс называется прямым базовым, если он находится в списке-базовых, и косвенным базовым, если сам не являясь прямым базовым, он служит базовым для одного из классов списка-базовых.
Отметим, что в обозначении имя-класса :: имя конструкция, имя может быть именем члена косвенного базового класса. Такое обозначение просто указывает класс, в котором следует начинать поиск этого имени.
Приведем пример:
class A { public: void f(); }
class B: public A {};
class C: public B { public: void f(); }
void C::f()
{
f(); // вызов f() из C
A::f(); // вызов f() из A
B::f(); // вызов f() из A
}
Здесь дважды вызывается A::f(), поскольку это единственная функция f() в классе B.
Инициализация объектов, представляющих базовые классы, задается в конструкторах, см. §R.12.6.2.
R.10.1 Множественные базовые классы
Класс может быть производным по отношению к любому числу базовых классов. Приведем пример:
class A {/*… */};
class B {/*… */};
class C {/*… */};
class D: public A, public B, public C {/*… */};
Использование более, чем одного прямого базового класса называется множественным наследованием.
Порядок наследования не важен, если не учитывать вопросов, связанных со стандартной инициализацией с помощью конструктора (§R.12.1), уничтожением (§R.12.4) и размещением в памяти ($$r.5.4, §R.9.2, §R.11.1). Порядок выделения памяти для базовых классов определяется реализацией.
Нельзя указывать класс в качестве прямого базового по отношению к производному классу более одного раза, но косвенным базовым классом он может быть неоднократно.
class B {/*… */};
class D: public B, public B {/*… */}; // недопустимо
class L {/*… */};
class A: public L {/*… */};
class B: public L {/*… */};
class C: public A, public B {/*… */}; // нормально
Здесь объект класса C будет иметь два вложенных объекта класса L.
К спецификации базового класса можно добавить служебное слово virtual. Отдельный объект виртуального базового класса V разделяется между всеми базовыми классами, которые указали V при задании своих базовых классов. Приведем пример:
class V {/*… */};
class A: virtual public V {/*… */};
class B: virtual public V {/*… */};
class C: public A, public B {/*… */};
Здесь объект класса C будет иметь только один вложенный объект класса V.
Класс может содержать виртуальные и невиртуальные базовые классы одного типа, например:
class B {/*… */};
class X: virtual public B {/*… */};
class Y: virtual public B {/*… */};
class Z: public B {/*… */};
class AA: public X, public Y, public Z {/*… */};
Здесь объект класса AA будет иметь два вложенных объекта класса B: из класса Z и виртуальный, разделяемый между классами X и Y.
R.10.1.1 Неоднозначности
Доступ к базовому классу должен быть задан однозначно. Доступ к члену базового класса считается неоднозначным, если выражение, используемое для доступа, задает более одной функции, объекта, типа или элемента перечисления. Проверка на однозначность происходит до проверки возможности доступа (§R.11). Приведем пример:
class A {
public:
int a;
int (*b)();
int f();
int f(int);
int g();
};
class B {
int a;
int b();
public:
int f();
int g();
int h();
int h(int);
};
class C: public A, public B {};
void g(C* pc)
{
pc-›a = 1; // ошибка: неоднозначность: A::a или B::a
pc-›b(); // ошибка: неоднозначность: A::b или B::b
pc-›f(); // ошибка: неоднозначность: A::f или B::f
pc-›f(1); // ошибка: неоднозначность: A::f или B::f
pc-›g(); // ошибка: неоднозначность: A::g или B::g
pc-›g = 1; // ошибка: неоднозначность: A::g или B::g
pc-›h(); // нормально
pc-›h(1); // нормально
}
Если имя перегруженной функции установлено однозначно, то прежде проверки возможности доступа происходит еще и разрешение перегрузки. Неоднозначность можно устранить, уточняя используемое имя именем класса, например, так:
class A {
public:
int f();
};
class B {
public:
int f();
};
class C: public A, public B {
int f() { return A::f() + B::f(); }
};
Если используются виртуальные базовые классы, до отдельной функции, объекта, типа или элемента перечисления можно добраться несколькими путями, двигаясь по направленному ацикличному графу, который образуют базовые классы. Но это не является неоднозначностью. Идентичное же использование невиртуальных базовых классов порождает неоднозначность, поскольку в этом случае участвует в задании доступа более одного вложенного объекта. Приведем пример:
class V { public: int v; };
class A { public: int a; };
class B: public A, public virtual V {};
class C: public A, public virtual V {};
class D: public B, public C { public: void f(); };
void D::f()
{
Интервал:
Закладка: