Множественное наследование
Если один класс наследуется от другого класса, первый класс называется подклассом класса, а второй класс — суперклассом первого. Каждый определенный пользователем класс должен иметь хотя бы один прямой суперкласс, т. е. по крайней мере один класс должен появиться в части is-a конструктора defclass. Множественное наследование происходит, когда класс имеет более одного суперкласса. В этом случае COOL проверяет список прямых суперклассов для нового класса и устанавливает некоторый линейный порядок, называемый списком предшествования классов. Новый класс наследует слоты и обработчики сообщений каждого класса в списке предшествования классов. Слоты и обработчики сообщений класса, стоящего в начале списка предшествования, переопределяют конфликтующие определения из других классов, ранее найденных в списке предшествования. Класс, стоящий перед другим классом в списке предшествования, называется более определенным. Все списки предшествования классов оканчиваются системным классом object, и большинство (если не все) списков предшествования классов для определенных пользователем классов заканчиваются парой системных классов — user и object. Вывести на экран список предшествования можно с помощью функции describe-class (см. гл. 15). Для определения списка предшествования классов нового класса COOL использует иерархию наследования от непосредственного суперкласса. COOL рекурсивно применяет следующие два правила к непосредственным суперклассам класса, список предшествования которого необходимо определить: 1. Класс имеет более высокий приоритет, чем любой из его суперклассов. 2. Определение класса задает приоритет между его непосредственными суперклассами. В случае если, воспользовавшись приведенными выше правилами, можно построить более одного списка предшествования, то COOL построит наиболее простой список, получаемый с помощью обхода дерева наследования класса в глубину. Эта эвристика максимально долго сохраняет родственное дерево класса. Например, если ребенок наследует основные черты от матери и отца, а мать и отец в свою очередь наследуют черты от своих родителей, список предшествования классов для ребенка будет следующим: ребенок, мать, мать матери, отец матери, отец, мать отца, отец отца. Возможны и другие варианты создания списка предшествования, удовлетворяющие правилам, например: ребенок, мать, отец, отец отца, мать матери, мать отца, отец матери. Однако COOL воспользуется первым вариантом, т. к. он сохраняет фамильное дерево класса настолько долго, насколько это возможно. Приведем несколько примеров, иллюстрирующих построение списка предшествования класса. Пример 11.1. Класс А
(defclass A (is-a USER)) Класс а наследует свои свойства непосредственно от класса user. Список предшествования классов для a: a user object. Пример 11.2. Класс В (defclass В (is-a USER)) Класс в также наследует свои свойства непосредственно от класса USER. Список предшествования классов для B: B USER OBJECT. Пример 11.3. Класс C (defclass С (is-a А В)) Класс C наследует свои свойства от классов A и B. Список предшествования классов для C: C A B USER OBJECT.
Пример 11.4. Класс D (defclass D (is-a В А)) Класс D наследует свои свойства от классов B и A. Список предшествования классов для D: D B A USER OBJECT. Пример 11.5. Класс Е (defclass E (is-a А С)) По правилу 2 класс A должен предшествовать C. Однако C является подклассом A и не может следовать за A в списке предшествования классов без нарушения правила 1. Таким образом, подобное определение класса E является ошибкой. Пример 11.6. Допустимое определение класса E (defclass E (is-a С А)) Указание класса A в списке суперклассов класса E в данном случае является излишним, т. к. класс C является наследником A. Однако такое определение класса E не нарушает правил, используемых для определения списка предшествования классов, и допустимо. Список предшествования классов для E: Е С А В USER OBJECT.
|