Язык программирования C++ для профессионалов


Виртуальные базовые классы - часть 2


Каждый производный класс добавляет новые свойства окна. Чтобы воспользоваться комбинацией всех этих свойств, мы должны гарантировать, что один и тот же объект класса window используется для представления вхождений базового класса window в эти производные классы. Именно это обеспечивает описание window во всех производных классах как виртуального базового класса.

Можно следующим образом изобразить состав объекта класса window_w_border_and_menu:

Чтобы увидеть разницу между обычным и виртуальным наследованием, сравните этот рисунок с рисунком из §6.5, показывающим состав объекта класса satellite. В графе наследования каждый базовый класс с данным именем, который был указан как виртуальный, будет представлен единственным объектом этого класса. Напротив, каждый базовый класс, который при описании наследования не был указан как виртуальный, будет представлен своим собственным объектом.

Теперь надо написать все эти функции draw(). Это не слишком трудно, но для неосторожного программиста здесь есть ловушка. Сначала пойдем самым простым путем, который как раз к ней и ведет:

void window_w_border::draw() { window::draw(); // рисуем рамку }

void window_w_menu::draw() { window::draw(); // рисуем меню }

Пока все хорошо. Все это очевидно, и мы следуем образцу определения таких функций при условии единственного наследования, который работал прекрасно. Однако, в производном классе следующего уровня появляется ловушка:

void window_w_border_and_menu::draw() // ловушка! { window_w_border::draw(); window_w_menu::draw();

// теперь операции, относящиеся только // к окну с рамкой и меню }

На первый взгляд все вполне нормально. Как обычно, сначала выполняются все операции, необходимые для базовых классов, а затем те, которые относятся собственно к производным классам. Но в результате функция window::draw() будет вызываться дважды! Для большинства графических программ это не просто излишний вызов, а порча картинки на экране. Обычно вторая выдача на экран затирает первую.

Чтобы избежать ловушки, надо действовать не так поспешно. Мы отделим действия, выполняемые базовым классом, от действий, выполняемых из базового класса. Для этого в каждом классе введем функцию _draw(), которая выполняет нужные только для него действия, а функция draw() будет выполнять те же действия плюс действия, нужные для каждого базового класса. Для класса window изменения сводятся к введению излишней функции:




- Начало -  - Назад -  - Вперед -