В приведенных примерах оба родителя играли симметричные роли, но это не всегда так. Иногда вклад каждого из них различен по своей природе.
Важным приложением множественного наследования является обеспечение реализации абстракции, описанной отложенным классом, используя свойства, обеспечиваемые эффективным классом. Один класс абстрактен, второй - эффективен.
Рассмотрим реализацию стека, заданную массивом. У нас уже есть классы для поддержки стеков и массивов в отдельности (абстрактный STACK и эффективный ARRAY, см. предыдущие лекции). Лучший способ реализации класса ARRAYED_STACK (стек, заданный массивом) - описать его как наследника классов STACK и ARRAY. Это концептуально верно: стек-массив одновременно является стеком (с точки зрения клиента) и массивом (с позиций поставщика). Вот описание класса:
indexing description: "Стек, реализованный массивом" class ARRAYED_STACK [G] inherit STACK [G] ARRAY [G] ... Здесь будут добавлены предложения переименования ... feature ...Реализация отложенных подпрограмм класса STACK в терминах операций класса ARRAY (см. ниже)... end
ARRAYED_STACK предлагает ту же функциональность, что и STACK, делая эффективными отложенные компоненты: full, put, count ..., реализуя их как операции над массивом.
Вот схема некоторых типичных компонентов: full, count и put. Так, условие, при котором стек полон, имеет вид:
full: BOOLEAN is -- Является ли стек (его представление) заполненным? do Result := (count = capacity) end
Компонент capacity унаследован от класса ARRAY и задает емкость стека, равную числу элементов массива. Для count потребуется ввести атрибут:
count: INTEGER
Это пример эффективной реализации отложенного компонента как атрибута. Наконец,
put (x: G) is -- Втолкнуть x на вершину. require not full do count := count + 1 array_put (x, count) end
Процедура array_put унаследована от класса ARRAY. Ее цель - записать новое значение в указанный элемент массива.
Компоненты capacity и array_put имели в классе ARRAY имена count и put. Смену прежних имен мы поясним позднее. |