Множественное наследование




Если класс A является базовым классом для B, то B наследует атрибутыA. т.е. B содержит A плюс еще что-то. С учетом этого становится очевидно,что хорошо, когда класс B может наследовать из двух базовых классов A1 иA2. Это называется множественным наследованием. Приведем некий типичный пример множественного наследования. Пусть естьдва библиотечных класса displayed и task. Первый представляет задачи,информация о которых может выдаваться на экран с помощью некоторогомонитора, а второй - задачи, выполняемые под управлением некоторогодиспетчера. Программист может создавать собственные классы, например,такие: class my_displayed_task: public displayed, public task { // текст пользователя }; class my_task: public task { // эта задача не изображается // на экране, т.к. не содержит класс displayed // текст пользователя }; class my_displayed: public displayed { // а это не задача // т.к. не содержит класс task // текст пользователя }; Если наследоваться может только один класс, то пользователю доступнытолько два из трех приведенных классов. В результате либо получаетсядублирование частей программы, либо теряется гибкость, а, как правило,происходит и то, и другое. Приведенный пример проходит в С++ безо всякихдополнительных расходов времени и памяти по сравнению с программами, вкоторых наследуется не более одного класса. Статический контроль типов отэтого тоже не страдает. Все неоднозначности выявляются на стадии трансляции: class task { public: void trace (); //... }; class displayed { public: void trace (); //... }; class my_displayed_task:public displayed, public task { // в этом классе trace () не определяется }; void g (my_displayed_task * p) { p -> trace (); // ошибка: неоднозначность } В этом примере видны отличия С++ от объектно-ориентированных диалектовязыка Лисп, в которых есть множественное наследование. В этих диалектахнеоднозначность разрешается так: или считается существенным порядокописания, или считаются идентичными объекты с одним и тем же именем вразных базовых классах, или используются комбинированные способы, когдасовпадение объектов доля базовых классов сочетается с более сложнымспособом для производных классов. В С++ неоднозначность, как правило,разрешается введением еще одной функции: class my_displayed_task:public displayed, public task { //... public: void trace () { // текст пользователя displayed::trace (); // вызов trace () из displayed task::trace (); // вызов trace () из task } //... }; void g (my_displayed_task * p) { p -> trace (); // теперь нормально }

Инкапсуляция

Пусть члену класса (неважно функции-члену или члену, представляющемуданные) требуется защита от "несанкционированного доступа". Как разумноограничить множество функций, которым такой член будет доступен? Очевидныйответ для языков, поддерживающих объектно-ориентированноепрограммирование, таков: доступ имеют все операции, которые определены дляэтого объекта, иными словами, все функции-члены. Например: class window { //... protected: Rectangle inside; //... }; class dumb_terminal: public window { //... public: void prompt (); //... }; Здесь в базовом классе window член inside типа Rectangle описываетсякак защищенный (protected), но функции-члены производных классов,например, dumb_terminal::prompt(), могут обратиться к нему и выяснить, скакого вида окном они работают. Для всех других функций членwindow::inside недоступен. В таком подходе сочетается высокая степень защищенности(действительно, вряд ли вы "случайно" определите производный класс) сгибкостью, необходимой для программ, которые создают классы и используютих иерархию (действительно, "для себя" всегда можно в производных классахпредусмотреть доступ к защищенным членам). Неочевидное следствие из этого: нельзя составить полный иокончательный список всех функций, которым будет доступен защищенный член,поскольку всегда можно добавить еще одну, определив ее как функцию-член вновом производном классе. Для метода абстракции данных такой подход частобывает мало приемлемым. Если язык ориентируется на метод абстракцииданных, то очевидное для него решение - это требование указывать вописании класса список всех функций, которым нужен доступ к члену. В С++для этой цели используется описание частных (private) членов. Оноиспользовалось и в приводившихся описаниях классов complex и shape. Важность инкапсуляции, т.е. заключения членов в защитную оболочку,резко возрастает с ростом размеров программы и увеличивающимся разбросомобластей приложения. В $$6.6 более подробно обсуждаются возможности языкапо инкапсуляции.

Пределы совершенства

Язык С++ проектировался как "лучший С", поддерживающий абстракциюданных и объектно-ориентированное программирование. При этом он долженбыть пригодным для большинства основных задач системного программирования. Основная трудность для языка, который создавался в расчете на методыупрятывания данных, абстракции данных и объектно-ориентированногопрограммирования, в том, что для того, чтобы быть языком общегоназначения, он должен: - идти на традиционных машинах; - сосуществовать с традиционными операционными системами и языками; - соперничать с традиционными языками программирования в эффективностивыполнения программы; - быть пригодным во всех основных областях приложения. Это значит, что должны быть возможности для эффективных числовыхопераций (арифметика с плавающей точкой без особых накладных расходов,иначе пользователь предпочтет Фортран) и средства такого доступа к памяти,который позволит писать на этом языке драйверы устройств. Кроме того, надоуметь писать вызовы функций в достаточно непривычной записи, принятой дляобращений в традиционных операционных системах. Наконец, должна бытьвозможность из языка, поддерживающего объектно-ориентированноепрограммирование, вызывать функции, написанные на других языках, а издругих языков вызывать функцию на этом языке, поддерживающемобъектно-ориентированное программирование. Далее, нельзя рассчитывать на широкое использование искомого языкапрограммирования как языка общего назначения, если реализация его целикомполагается на возможности, которые отсутствуют в машинах с традиционнойархитектурой. Если не вводить в язык возможности низкого уровня, то придется дляосновных задач большинства областей приложения использовать некоторыеязыки низкого уровня, например С или ассемблер. Но С++ проектировался срасчетом, что в нем можно сделать все, что допустимо на С, причем безувеличения времени выполнения. Вообще, С++ проектировался, исходя изпринципа, что не должно возникать никаких дополнительных затрат времени ипамяти, если только этого явно не пожелает сам программист. Язык проектировался в расчете на современные методы трансляции,которые обеспечивают проверку согласованности программы, ее эффективностьи компактность представления. Основным средством борьбы со сложностьюпрограмм видится, прежде всего, строгий контроль типов и инкапсуляция.Особенно это касается больших программ, создаваемых многими людьми.Пользователь может не являться одним из создателей таких программ, и можетвообще не быть программистом. Поскольку никакую настоящую программунельзя написать без поддержки библиотек, создаваемых другимипрограммистами, последнее замечание можно отнести практически ко всемпрограммам. С++ проектировался для поддержки того принципа, что всякая программаесть модель некоторых существующих в реальности понятий, а класс являетсяконкретным представлением понятия, взятого из области приложения ($$12.2).Поэтому классы пронизывают всю программу на С++, и налагаются жесткиетребования на гибкость понятия класса, компактность объектов класса иэффективность их использования. Если работать с классами будет неудобноили слишком накладно, то они просто не будут использоваться, и программывыродятся в программы на "лучшем С". Значит пользователь не сумеетнасладиться теми возможностями, ради которых, собственно, и создавалсяязык.

* ГЛАВА 2. ОПИСАНИЯ И КОНСТАНТЫ

"Совершенство достижимо только в момент краха". (С.Н. Паркинсон) В данной главе описаны основные типы (char, int, float и т.д.) и способы построения на их основе новых типов (функций, векторов, указателей и т.д.). Описание вводит в программу имя, указав его тип и, возможно, начальное значение. В этой главе вводятся такие понятия, как описание и определение, типы, область видимости имен, время жизни объектов. Даются обозначения литеральных констант С++ и способы задания символических констант. Приводятся примеры, которые просто демонстрируют возможности языка. Более осмысленные примеры, иллюстрирующие возможности выражений и операторов языка С++, будут приведены в следующей главе. В этой главе лишь упоминаются средства для определения пользовательских типов и операций над ними. Они обсуждаются в главах 5 и 7.

ОПИСАНИЯ

Имя (идентификатор) следует описать прежде, чем оно будет использоваться в программе на С++. Это означает, что нужно указать его тип, чтобы транслятор знал, к какого вида объектам относится имя. Ниже приведенынесколько примеров, иллюстрирующих все разнообразие описаний: char ch; int count = 1; char* name = "Njal"; struct complex { float re, im; }; complex cvar; extern complex sqrt(complex); extern int error_number; typedef complex point; float real(complex* p) { return p->re; }; const double pi = 3.1415926535897932385; struct user; template<class T> abs(T a) { return a<0? -a: a; } enum beer { Carlsberg, Tuborg, Thor }; Из этих примеров видно, что роль описаний не сводится лишь к привязкетипа к имени. Большинство указанных описаний одновременно являютсяопределениями, т.е. они создают объект, на который ссылается имя. Для ch, count, name и cvar таким объектом является элемент памяти соответствующего размера. Этот элемент будет использоваться как переменная, и говорят, что для него отведена память. Для real подобным объектом будет заданная функция. Для константы pi объектом будет число 3.1415926535897932385. Для complex объектом будет новый тип. Для point объектом является тип complex, поэтому point становится синонимом complex. Следующие описания уже не являются определениями: extern complex sqrt(complex); extern int error_number; struct user; Это означает, что объекты, введенные ими, должны быть определены где-то в другом месте программы. Тело функции sqrt должно быть указано в каком-то другом описании. Память для переменной error_number типа int должна выделяться в результате другого описания error_number. Должно быть и какое-то другое описание типа user, из которого можно понять, что это за тип. В программе на языке С++ должно быть только одно определение каждого имени, но описаний может быть много. Однако все описания должны быть согласованы по типу вводимого в них объекта. Поэтому в приведенном ниже фрагменте содержатся две ошибки: int count; int count; // ошибка: переопределение extern int error_number; extern short error_number; // ошибка: несоответствие типов Зато в следующем фрагменте нет ни одной ошибки (об использовании extern см. #4.2): extern int error_number; extern int error_number; В некоторых описаниях указываются "значения" объектов, которые ониопределяют: struct complex { float re, im; }; typedef complex point; float real(complex* p) { return p->re }; const double pi = 3.1415926535897932385; Для типов, функций и констант "значение" остается неизменным;для данных, не являющихся константами, начальное значение можетвпоследствии изменяться: int count = 1; char* name = "Bjarne"; //... count = 2; name = "Marian"; Из всех определений только следующее не задает значения: char ch; Всякое описание, которое задает значение, является определением.

Область видимости

Описанием определяется область видимости имени. Это значит, чтоимя может использоваться только в определенной части текста программы.Если имя описано в функции (обычно его называют "локальным именем"), тообласть видимости имени простирается от точки описаниядо конца блока, в котором появилось это описание. Если имя не находитсяв описании функции или класса (его обычно называют "глобальным именем"),то область видимости простирается от точки описания до конца файла,в котором появилось это описание.Описание имени в блоке может скрывать описание в объемлющем блоке илиглобальное имя; т.е. имя может быть переопределено так, что оно будетобозначать другой объект внутри блока. После выхода из блока прежнеезначение имени (если оно было) восстанавливается. Приведем пример: int x; // глобальное x void f(){ int x; // локальное x скрывает глобальное x x = 1; // присвоить локальному x { int x; // скрывает первое локальное x x = 2; // присвоить второму локальному x } x = 3; // присвоить первому локальному x} int* p = &x; // взять адрес глобального x В больших программах не избежать переопределения имен. К сожалению,человек легко может проглядеть такое переопределение. Возникающиеиз-за этого ошибки найти непросто, возможно потому, что онидостаточно редки. Следовательно, переопределение имен следуетсвести к минимуму. Если вы обозначаете глобальные переменные илилокальные переменные в большой функции такими именами, как i или x,то сами напрашиваетесь на неприятности. Есть возможность с помощью операции разрешения области видимости:: обратиться к скрытому глобальному имени, например: int x; void f2() { int x = 1; // скрывает глобальное x::x = 2; // присваивание глобальному x } Возможность использовать скрытое локальное имя отсутствует. Область видимости имени начинается в точке его описания (поокончании описателя, но еще до начала инициализатора - см. $$R.3.2). Этоозначает, что имя можно использовать даже до того, как задано егоначальное значение. Например: int x; void f3() { int x = x; // ошибочное присваивание } Такое присваивание недопустимо и лишено смысла. Если вы попытаетесь транслировать эту программу, то получите предупреждение: "использование до задания значения". Вместе с тем, не применяя оператора::, можно использовать одно и то же имя для обозначения двух различных объектов блока. Например: int x = 11; void f4() // извращенный пример { int y = x; // глобальное x int x = 22; y = x; // локальное x } Переменная y инициализируется значением глобального x, т.е. 11, а затем ей присваивается значение локальной переменной x, т.е. 22. Имена формальных параметров функции считаются описанными в самом большом блоке функции, поэтому в описании ниже есть ошибка: void f5(int x) { int x; // ошибка } Здесь x определено дважды в одной и той же области видимости. Это хотя и не слишком редкая, но довольно тонкая ошибка.

Объекты и адреса

Можно выделять память для "переменных", не имеющих имен, и использовать эти переменные. Возможно даже присваивание таким странно выглядящим "переменным", например, *p[a+10]=7. Следовательно, есть потребность именовать "нечто хранящееся в памяти". Можно привести подходящую цитату из справочного руководства: "Любой объект - это некоторая область памяти, а адресом называется выражение, ссылающееся на объект или функцию" ($$R.3.7). Слову адрес (lvalue - left value, т.е. величина слева) первоначально приписывался смысл "нечто, что может в присваивании стоять слева". Адрес может ссылаться и на константу (см. $$2.5). Адрес, который не был описан со спецификацией const, называется изменяемым адресом.

Время жизни объектов

Если только программист не вмешается явно, объект будет создан при появлении его определения и уничтожен, когда исчезнет из области видимости. Объекты с глобальными именами создаются, инициализируются (причем только один раз) и существуют до конца программы. Если локальные объекты описаны со служебным словом static, то они также существуют до конца программы. Инициализация их происходит, когда в первый раз управление "проходит через" описание этих объектов, например: int a = 1; void f() { int b = 1; // инициализируется при каждом вызове f() static int c = a; // инициализируется только один раз cout << " a = " << a++ << " b = " << b++ << " c = " << c++ << '\n'; } int main() { while (a < 4) f(); } Здесь программа выдаст такой результат: a = 1 b = 1 c = 1 a = 2 b = 1 c = 2 a = 3 b = 1 c = 3 ''Из примеров этой главы для краткости изложения исключенамакрокоманда #include <iostream>. Она нужна лишь в тех из них, которыевыдают результат. Операция "++" является инкрементом, т. е. a++ означает: добавить 1 к переменной a. Глобальная переменная или локальная переменная static, которая не была явно инициализирована, инициализируется неявно нулевым значением (#2.4.5). Используя операции new и delete, программист может создавать объекты, временем жизни которых он управляет сам (см. $$3.2.6).

ИМЕНА

Имя (идентификатор) является последовательностью букв или цифр. Первый символ должен быть буквой. Буквой считается и символ подчеркивания _. Язык С++ не ограничивает число символов в имени. Но в реализацию входят программные компоненты, которыми создатель транслятора управлять не может (например, загрузчик), а они, к сожалению, могут устанавливать ограничения. Кроме того, некоторые системные программы, необходимые для выполнения программы на С++, могут расширять или сужать множество символов, допустимых в идентификаторе. Расширения (например, использование $ в имени) могут нарушить переносимость программы. Нельзя использовать в качестве имен служебные слова С++ (см. $$R.2.4), например: hello this_is_a_most_unusially_long_name DEFINED foO bAr u_name HorseSense var0 var1 CLASS _class ___ Теперь приведем примеры последовательностей символов, которые не могутиспользоваться как идентификаторы: 012 a fool $sys class 3var pay.due foo~bar.name if Заглавные и строчные буквы считаются различными, поэтому Count иcount - разные имена. Но выбирать имена, почти не отличающиесядруг от друга, неразумно. Все имена, начинающиеся с символаподчеркивания, резервируются для использования в самой реализацииили в тех программах, которые выполняются совместно с рабочей,поэтому крайне легкомысленно вставлять такие имена всвою программу. При разборе программы транслятор всегда стремится выбрать самуюдлинную последовательность символов, образующих имя, поэтому var10- это имя, а не идущие подряд имя var и число 10. По той же причинеelseif - одно имя (служебное), а не два служебных имени else и if.

ТИПЫ

С каждым именем (идентификатором) в программе связан тип. Онзадает те операции, которые могут применяться к имени (т.е. к объекту,который обозначает имя), а также интерпретацию этих операций.Приведем примеры: int error_number; float real(complex* p); Поскольку переменная error_number описана как int (целое), ей можноприсваивать, а также можно использовать ее значения в арифметическихвыражениях. Функцию real можно вызывать с параметром, содержащимадрес complex. Можно получать адреса и переменной, и функции.Некоторые имена, как в нашем примере int и complex, являются именамитипов. Обычно имя типа нужно, чтобы задать в описании типа некотороедругое имя. Кроме того, имя типа может использоватьсяв качестве операнда в операциях sizeof (с ее помощью определяютразмер памяти, необходимый для объектов этого типа) и new (с еепомощью можно разместить в свободной памяти объект этого типа).Например: int main() { int* p = new int; cout << "sizeof(int) = " << sizeof(int) '\n'; } Еще имя типа может использоваться в операции явного преобразованияодного типа к другому ($$3.2.5), например: float f; char* p; //... long ll = long(p); // преобразует p в long int i = int(f); // преобразует f в int

Основные типы



Поделиться:




Поиск по сайту

©2015-2025 poisk-ru.ru
Все права принадлежать их авторам. Данный сайт не претендует на авторства, а предоставляет бесплатное использование.
Дата создания страницы: 2016-02-16 Нарушение авторских прав и Нарушение персональных данных


Поиск по сайту: