|
Программирование >> Дополнительные возможности наследования
в листинге 15.6 был изменен интерфейс класса PartsCatalog и переписана функция main(). Интерфейсы других классов остались такими же, как и в листинге 15.5. В строке 258 листинга 15.6 класс PartsCatalog производится как private от класса PartsList. Интерфейс класса PartsCatalog остался таким же, как и в листинге 15.5, хотя, конечно же, необходимость в объектах класса PartsList как переменных-членах отпала. Функция-член ShowAlK) класса PartsCatalog вызывает функцию IterateO из PartsList, параметром которой задается указатель на функцию-член класса Part. Таким образом, функция ShowAll() выполняет роль открытого интерфейса, позволяя пользователям получать информацию, не обращаясь напрямую к закрытой функции IterateO, прямой доступ к которой закрыт для клиентов класса PartsCatalog. Функция Insert()тoжe изменилась. Обратите внимание, в строке 274 функция FindO теперь вызывается непосредственно, поскольку она наследуется из базового класса. Чтобы при вызове функции InsertO не возникло зацикливания функции на самое себя, в строке 275 делается явный вызов функции с указанием имени класса. Таким образом, если методам класса PartsCatalog необходимо вызвать методы PartsList, они могут делать это напрямую. Единствейное исключение состоит в том, что при необходимости заместить метод базового класса в классе PartsList следует явно указать класс и имя функции. Закрытое наследование позволяет PartsCatalog унаследовать функциональность базового класса и создавать интерфейс, косвенно открывающий доступ к его методам, которые нельзя вызывать напрямую. Рекомендуется Применяйте открытое наследование, когда производный класс является разновидностью базового. Используйте вложение классов, когда необходимо делегировать выполнение задач другому классу, ограничив при этом доступ к его защищенным членам. Применяйте закрытое наследование, если необходимо реализовать один класс в пределах другого и обеспечить доступ к защищенным членам базового класса. Не рекомендуется Не применяйте закрытое наследование, если необходимо использовать более одного объекта базового класса. Для этих целей больше подойдет вложение классов. Например, если для одного объекта PartsCatalog необходимы два объекта PartsList. вы не сможете использовать закрытое наследование. Не используйте открытое наследование, если необходимо закрыть клиентам производного класса прямой доступ к методам базового класса. Классы-друзья Иногда для выполнения задач, поставленных перед программой, необходимо обеспечить взаимодействие нескольких независимых классов. Например, классы PartNode и PartsList тесно взаимосвязаны, и было бы удобно, если бы в PartsList можно было напрямую использовать указатель itsPart класса PartNode. Конечно, можно было бы объявить itsPart как открытую или хотя бы защищенную переменную-член, но это далеко не лучший путь, противоречащий самой идее использования классов. Поскольку указатель itsPart является специфическим членом класса PartNode, его следует оставить недоступным для внешних классов. Однако, если вы хотите предоставить данные или закрытые методы какому-либо иному классу, достаточно объявить этот класс другом. Это расширит интерфейс вашего класса возможностями класса-друга. После того как в PartsNode класс PartsList будет объявлен другом, переменные-члены и методы класса PartsNode станут доступными для PartsList. Важно заметить, что дружественность класса не передается на другие классы. Иными словами, если вы мой друг, а Ваня - ваш друг, это вовсе не значит, что Ваня также и мой друг. Кроме того, дружба не наследуется. Опять же, хотя вы мой друг и я хочу рассказать вам свои секреты, это не означает, что я желаю поделиться ими с вашими детьми. Наконец, дружественность классов односторонняя. Объявление одного класса другом какого-либо иного класса не делает последний другом первого. Вы при желании может поделиться своими секретами со мной, но это не значит, что я должен рассказать вам свои секреты. В листинге 15.7 представлена версия листинга 15.6, в которой используется объявление класса друга. Так, класс PartsList объявляется как друг класса PartNode. Еще раз напомним, что это объявление не делает класс PartNode другом класса PartsList. Яввяинг 15.7. Нсшьшаннв кхассвв-друзвО 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 Sinclude <iostream.h> ... *. Класс Part ** ** ** ** Абстрактный базовый класс всех деталей class Part { public: Part():itsPartNumber(1) { } Part(int PartNumber): itsPartNumber(PartNumber){ } virtual Part(){ } int GetPartNumber() const { return itsPartNumber; } virtual void DisplayO const =0: private: int ItsPartNumber; } ; выполнение чистой виртуальной функции в стандартном виде для всех производных классов void Part;:Display() const cout \ nPart Number: ; cout itsPartNumber endl; > .............. Класс Gar Part ............ 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 class CarPart : public Part { public: CarPart():itsModelYear(94){ } CarPart(int year, int partNumber); virtual void DisplayO const { Part::Display(); cout Model Year: ; cout ItsModelYear << endl; private: int itsModelYear; } : CarPart::CarPart(int year, int partNumber): itsModelYear(year), Part(partNumber) { } ........... Класс AirPlane Part class AirPlanePart : public Part. { public: AirPlanePart():itsEngineNumber(1){ } ; AirPlanePart (int EngineNumber, int PartNumber); virtual void DisplayO const { Part: :DisplayO; cout Engine No,: ; cout << ItsEngineNumber << endl; private: int ItsEngineNumber; AirPlanePart::AirPlanePart (int EngineNumber, int PartNumber): ItsEngineNumber(EngineNumber), Part(PartNumber) { } .**..... ... .. Класс Part Node * > class PartNode public: friend class PartsList;
|
© 2006 - 2025 pmbk.ru. Генерация страницы: 0
При копировании материалов приветствуются ссылки. |