ЭЛЕКТРОННАЯ БИБЛИОТЕКА КОАПП
Сборники Художественной, Технической, Справочной, Английской, Нормативной, Исторической, и др. литературы.



 

Часть 3

                                                                              
                                  Глава 1 Турне по С++
           
                                       Единственный способ  изучать новый язык
                                  программирования - писать на нем программы.
                                                            - Брайэн Керниган
           
             Эта глава  представляет  собой  краткий обзор основных черт языка
           программирования С++.  Сначала приводится программа на  С++,  затем
           показано, как ее откомпилировать и запустить, и как такая программа
           может выводить выходные данные и считывать входные.  В первой трети
           этой  главы  после  введения  описаны  наиболее  обычные черты С++:
           основные типы,  описания, выражения, операторы, функции и структура
           программы. Оставшаяся часть главы посвящена возможностям С++ по оп-
           ределению новых типов,  сокрытию  данных,  операциям,  определяемым
           пользователем, и иерархии определяемых пользователем типов.
                1.1 Введение
           
             Это турне проведет вас через ряд программ и  частей  программ  на
           С++. К концу у вас должно сложиться общее представление об основных
           особенностях С++,  и  будет  достаточно  информации,  чтобы  писать
           простые программы. Для точного и полного объяснения понятий, затро-
           нутых даже в самом маленьком законченном примере,  потребовалось бы
           несколько  страниц  определений.  Чтобы  не  превращать эту главу в
           описание или в обсуждение общих понятий,  примеры  снабжены  только
           самыми   короткими  определениями  используемых  терминов.  Термины
           рассматриваются позже,  когда будет больше примеров, способствующих
           обсуждению.
           
                1.1.1 Вывод
           
                 Прежде всего, давайте напишем программу, выводящую строку вы-
           дачи:
           
             #include 
           
             main() {
                 cout << "Hello, world\n";
             }
           
             Строка #include  сообщает компилятору, чтобы он включил
           стандартные возможности потока ввода и вывода,  находящиеся в файле
           stream.h.  Без этих описаний выражение cout << "Hello,  world\n" не
           имело бы смысла. Операция << ("поместить в"*) пишет свой первый ар-
           гумент во второй (в данном случае,  строку "Hello, world\n" в стан-
           дартный поток вывода cout).  Строка - это последовательность симво-
           лов,  заключенная в двойные кавычки. В строке символ обратной косой
           \,  за  которым следует другой символ,  обозначает один специальный
           символ,  в данном случае,  \n является символом новой строки. Таким
           образом выводимые символы состоят из Hello,  world и перевода стро-
           ки.
           
           ДДДДДДДДДДДДДДДДДДДД
           * Программирующим на C << известно как операция сдвига влево для
           целых. Такое использование << не утеряно,  просто в  дальнейшем  <<
           было определено для случая,  когда его левый операнд является пото-
           ком вывода. Как это делается, описано в #1.8. (прим. автора)
             Остальная часть программы
           
             main() { ... }
           
           определяет функцию,  названную main. Каждая программа должна содер-
           жать функцию с именем main,  и работа программы начинается с выпол-
           нения этой функции.
           
                1.1.2 Компиляция
           
             Откуда появились выходной поток cout и код,  реализующий операцию
           вывода <
           
             main() {
                 int inch = 0;  // inch - дюйм cout << "inches";  cin >> inch;
                 cout << inch;  cout << " in = "; cout << inch*2.54; cout << "
                 cm\n";
             }
           
             Первая строка функции main() описывает целую переменную inch.  Ее
           значение считывается с помощью операции >> ("взять из")  над  стан-
           дартным потоком ввода cin. Описания cin и >>, конечно же, находятся
           в . После ее запуска ваш терминал может выглядеть пример-
           но так:
           
             $ a.out
             inches=12
             12 in = 30.48 cm
             $
           
             В этом примере на каждую команду вывода приходится один оператор.
           Это  слишком  длинно.  Операцию  вывода  <<  можно  применять  к ее
           собственному результату,  так что последние четыре  команды  вывода
           можно было записать одним оператором:
           
             cout << inch << " in = " << inch*2.54 << " cm\n";
           
             В последующих  разделах  ввод и вывод будут описаны гораздо более
           подробно.  Вся эта глава фактически может рассматриваться как  объ-
           яснение того, как можно написать предыдущие программы на языке, ко-
           торый не обеспечивает операции ввода-вывода.  На самом деле, приве-
           денные  выше  программы  написаны на С++,  "расширенном" операциями
           ввода-вывода посредством использования библиотек и включения файлов
           с помощью #include. Другими словами, язык С++ в том виде, в котором
           он описан в справочном руководстве, не определяет средств ввода-вы-
           вода. Вместо этого исключительно с помощью средств, доступных любо-
           му программисту, определены операции << и >>.
                     1.2 Комментарии
           
             Часто бывает  полезно вставлять в программу текст,  который пред-
           назначается в качестве комментария только для  читающего  программу
           человека  и игнорируется компилятором в программе.  В С++ это можно
           сделать одним из двух способов.
           
             Символы /* начинают комментарий,  заканчивающийся  символами  */.
           Вся  эта  последовательность символов эквивалентна символу пропуска
           (например, символу пробела). Это наиболее полезно для многострочных
           комментариев и изъятия частей программы при редактировании,  однако
           следует помнить, что комментарии /* */ не могут быть вложенными.
           
             Символы // начинают комментарий,  который заканчивается  в  конце
           строки,  на  которой они появились.  Опять,  вся последовательность
           символов эквивалентна пропуску.  Этот способ наиболее  полезен  для
           коротких комментариев. Символы // можно использовать для того, что-
           бы закомментировать символы /* или */,  а символами /* можно заком-
           ментировать //.
                1.3 Типы и Описания
           
             Каждое имя и каждое выражение имеет тип,  определяющий  операции,
           которые могут над ними производиться. Например, описание
           
             int inch;
           
           определяет, что  inch имеет тип int,  то есть,  inch является целой
           переменной.
           
             Описание - это оператор, который вводит имя в программе. Описание
           задает  тип  этого  имени.  Тип определяет правильное использование
           имени или выражения. Для целых определены такие операции, как +, -,
           * и /. После того, как включен файл stream.h, объект типа int может
           также быть вторым операндом <<, когда первый операнд ostream.
           
             Тип объекта определяет не только то,  какие операции могут к нему
           применяться, но и смысл этих операций. Например, оператор
           
             cout << inch << " in = " << inch*2.54 << " cm\n";
           
           правильно обрабатывает  четыре  входных значения различным образом.
           Строки печатаются буквально, тогда как целое inch и значение с пла-
           вающей точкой inch*2.54 преобразуются из их внутреннего представле-
           ния в подходящее для человеческого глаза символьное представление.
           
             В С++ есть несколько основных типов и несколько  способов  созда-
           вать новые.  Простейшие виды типов С++ описываются в следующих раз-
           делах, а более интересные оставлены на потом.
           
                1.3.1 Основные Типы
           
             Основные типы,  наиболее непосредственно отвечающие средствам ап-
           паратного обеспечения, такие:
           
             char short int long float double
           
             Первые четыре типа используются для представления целых,  послед-
           ние два - для представления чисел с  плавающей  точкой.  Переменная
           типа char имеет размер, естественный для хранения символа на данной
           машине (обычно,  байт),  а переменная типа int имеет размер,  соот-
           ветствующий целой арифметике на данной машине (обычно, слово). Диа-
           пазон целых чисел,  которые могут быть представлены типом,  зависит
           от его размера (sizeof). В С++ размеры измеряются в единицах разме-
           ра данных типа char,  поэтому char по определению имеет размер еди-
           ница. Соотношение между основными типами можно записать так:
           
             1 =  sizeof(char) <= sizeof(short) <= sizeof(int) <= sizeof(long)
             sizeof(float) <= sizeof(double)
           
             В целом,  предполагать что-либо еще относительно  основных  типов
           неразумно.  В частности, то, что целое достаточно для хранения ука-
           зателя, верно не для всех машин.
             К основному  типу можно применять прилагательное const.  Это дает
           тип, имеющий те же свойства, что и исходный тип, за исключением то-
           го,  что  значение  переменных типа const не может изменяться после
           инициализации.
           
             const float pi = 3.14; const char plus = '+';
           
             Символ, заключенный  в  одинарные  кавычки,  является  символьной
           константой. Заметьте, что часто константа, определенная таким обра-
           зом, не занимает память. Просто там, где требуется, ее значение мо-
           жет использоваться непосредственно.  Константа должна инициализиро-
           ваться при описании. Для переменных инициализация необязательна, но
           настоятельно рекомендуется.  Оснований для введения локальной пере-
           менной без ее инициализации очень немного.
           
             К любой комбинации этих типов  могут  применяться  арифметические
           операции:
           
             + (плюс, унарный и бинарный)
             - (минус, унарный и бинарный)
             * (умножение)
             / (деление)
           
             А также операции сравнения:
           
             == (равно)
             != (не равно)
             < (меньше)
             > (больше)
             <= (меньше или равно)
             >= (больше или равно)
           
             Заметьте, что целое деление дает целый результат: 7/2 есть 3. Над
           целыми может выполняться операция % получения остатка: 7%2 равно 1.
           
              При присваивании и арифметических операциях  С++  выполняет  все
           осмысленные  преобразования между основными типами,  чтобы их можно
           было сочетать без ограничений:
           
             double d = 1; int i = 1; d = d + i; i = d + i;
           
                1.3.2 Производные Типы
           
             Вот операции, создающие из основных типов новые типы:
           
             * указатель на
             *const константный указатель на
             & ссылка на
             [] вектор* () функция, возвращающая
           
           ДДДДДДДДДДДДДДДДДДДД
           * одномерный массив. Это принятый термин (например, вектора
           прерываний), и  мы сочли,  что стандартный перевод его как "массив"
           затуманит изложение. (прим. перев.)
             Например:
           
             char* p  // указатель на символ char *const q // константный ука-
             затель на символ char v[10] // вектор из 10 символов
           
             Все вектора в качестве нижней границы индекса имеют ноль, поэтому
           в v десять элементов: v[0]..v[9]. Функции объясняются в #1.5, ссыл-
           ки в #1.9. Переменная указатель может содержать адрес объекта соот-
           ветствующего типа:
           
             char c;
             // ...
             p = &c; // p указывает на c
           
             Унарное & является операцией взятия адреса.
           
                1.4 Выражения и Операторы
           
             В С++ имеется богатый набор операций, с помощью которых в выраже-
           ниях образуются новые значения и  изменяются  значения  переменных.
           Поток  управления  в  программе  задается  с помощью операторов,  а
           описания используются для введения  в  программе  имен  переменных,
           констант и т.д. Заметьте, что описания являются операторами, поэто-
           му они свободно могут сочетаться с другими операторами.
                1.4.1 Выражения
           
             В С++  имеется  большое  число операций,  и они будут объясняться
           там, где (и если) это потребуется. Следует учесть, что операции
           
             ~ (дополнение)
             & (И)
             ^ (исключающее ИЛИ)
             | (включающее ИЛИ)
             << (логический сдвиг влево)
             >> (логический сдвиг вправо)
           
           применяются к  целым,  и  что  нет отдельного типа данных для логи-
           ческих действий.
           
             Смысл операции зависит от числа  операндов.  Унарное  &  является
           операцией взятия адреса, а бинарное & - это операция логического И.
           Смысл операции зависит также от типа ее операндов:  +  в  выражении
           a+b  означает сложение с плавающей точкой,  если операнды имеют тип
           float,  но целое сложение,  если они типа int.  В #1.8 объясняется,
           как  можно определить операцию для типа,  определяемого пользовате-
           лем,  без потери ее значения, предопределенного для основных и про-
           изводных типов.
           
             В С++  есть операция присваивания =,  а не оператор присваивания,
           как в некоторых языках.  Таким образом,  присваивание может  встре-
           чаться в неожиданном контексте, например, x=sqrt(a=3*x). Это бывает
           полезно.  a=b=c означает присвоение c объекту b, а затем объекту a.
           Другим  свойством операции присваивания является то,  что она может
           совмещаться с большинством бинарных операций.  Например,  x[i+3]*=4
           означает x[i+3]=x[i+3]*4,  за исключением того факта, что выражение
           x[i+3] вычисляется только один раз.  Это дает привлекательную  сте-
           пень  эффективности  без  необходимости  обращения к оптимизирующим
           компиляторам. К тому же это более кратко.
           
             В большинстве программ на С++ широко применяются указатели. Унар-
           ная операция * разыменовывает* указатель,  т.е.  *p есть объект, на
           который указывает p.  Эта операция также называется  косвенной  ад-
           ресацией. Например, если имеется char* p, то *p есть символ, на ко-
           торый указывает p.  Часто при работе с указателями  бывают  полезны
           операция  увеличения  ++ и операция уменьшения --.  Предположим,  p
           указывает на элемент вектора v,  тогда p++ делает p указывающим  на
           следующий элемент.
           
           ДДДДДДДДДДДДДДДДДДДД
           * англ. dereference - получить значение объекта, на который
           указывает данный указатель. (прим. перев.)
                1.4.2 Операторы Выражения
           
             Самый обычный вид оператора - выражение;.  Он состоит из  выраже-
           ния, за которым следует точка с запятой. Например:
           
             a = b*3+c; cout << "go go go"; lseek(fd,0,2);
           
                1.4.3 Пустой оператор
           
             Простейшей формой оператора является оператор:
           
             ;
           
             Он не делает ничего.  Однако он может быть полезен в тех случаях,
           когда синтаксис требует наличие оператора, а вам оператор не нужен.
           
                1.4.4 Блоки
           
             Блок - это возможно пустой список операторов,  заключенный в  фи-
           гурные скобки:
           
             { a=b+2; b++; }
           
           Блок позволяет рассматривать несколько операторов как один. Область
           видимости имени,  описанного в блоке,  простирается до конца блока.
           Имя  можно  сделать невидимым с помощью описаний такого же имени во
           внутренних блоках.
           
                1.4.5 Оператор if
           
             Программа в следующем примере осуществляет преобразование  дюймов
           в сантиметры и сантиметров в дюймы.  Предполагается, что вы укажете
           единицы измерения вводимых данных,  добавляя i для дюймов и  c  для
           сантиметров:
           
             #include 
           
             main() {
                 const float fac = 2.54; float x, in, cm; char ch = 0;
           
                 cout << "введите длину: "; cin >> x >> ch;
           
                 if (ch == 'i') { // inch - дюймы in = x; cm = x*fac;
                 }
                 else if (ch == 'c') // cm - сантиметры in = x/fac; cm = x;
                 }
                 else in = cm = 0;
           
                 cout << in << " in = " << cm << " cm\n";
             }
           
             Заметьте, что  условие  в  операторе  if  должно быть заключено в
           круглые скобки.
                1.4.6 Операторы switch
           
             Оператор switch  производит  сопоставление  значения с множеством
           констант. Проверки в предыдущем примере можно записать так:
           
             switch (ch) { case 'i':
                 in = x; cm = x*fac; break;
             case 'c':
                 in = x/fac; cm = x; break;
             default:
                 in = cm = 0; break;
             }
           
             Операторы break  применяются  для  выхода  из  оператора  switch.
           Константы в вариантах case должны быть различными, и если проверяе-
           мое значение не совпадает ни с одной из констант,  выбирается вари-
           ант default. Программисту не обязательно предусматривать default.
           
                1.4.7 Оператор while
           
             Рассмотрим копирование  строки,  когда  заданы  указатель p на ее
           первый символ и указатель q на целевую строку. По соглашению строка
           оканчивается символом с целым значением 0.
           
             while (p != 0) {
                 *q = *p; // скопировать символ
                 q = q+1; p = p+1;
             }
             *q = 0;      // завершающий символ 0 скопирован не был
           
             Следующее после  while  условие  должно  быть заключено в круглые
           скобки.  Условие вычисляется, и если его значение не ноль, выполня-
           ется непосредственно следующий за ним оператор.  Это повторяется до
           тех пор, пока вычисление условия не даст ноль.
           
             Этот пример слишком пространен.  Можно использовать  операцию  ++
           для непосредственного указания увеличения, и проверка упростится:
           
             while (*p) *q++ = *p++;
             *q = 0;
           
           где конструкция *p++ означает:  "взять символ, на который указывает
           p, затем увеличить p."
           
             Пример можно еще упростить,  так как указатель p разыменовывается
           дважды за каждый цикл.  Копирование символа можно делать тогда  же,
           когда производится проверка условия:
           
             while (*q++ = *p++) ;
           
             Здесь берется  символ,  на который указывает p,  p увеличивается,
           этот символ копируется туда,  куда указывает q,  и q увеличивается.
           Если символ ненулевой,  цикл повторяется.  Поскольку вся работа вы-
           полняется в условии,  не требуется ни одного оператора.  Чтобы ука-
           зать на это,  используется пустой оператор.  С++ (как и C) одновре-
           менно любят и ненавидят за возможность такого чрезвычайно  краткого
           ориентированного на выразительность программирования*.
           
           ДДДДДДДДДДДДДДДДДДДД
           * в оригинале expression-oriented (expression - выразительность и
           выражение). (прим. перев.)
                1.4.8 Оператор for
           
             Рассмотрим копирование десяти элементов одного вектора в другой:
           
             for (int i=0; i<10; i++) q[i]=p[i];
           
             Это эквивалентно
           
             int i = 0; while (i<10) {
                 q[i] = p[i]; i++;
             }
           
           но более удобочитаемо,  поскольку вся информация,  управляющая цик-
           лом,  локализована. При применении операции ++ к целой переменной к
           ней просто добавляется единица.  Первая часть оператора for не обя-
           зательно  должна  быть описанием,  она может быть любым оператором.
           Например:
           
             for (i=0; i<10; i++) q[i]=p[i];
           
           тоже эквивалентно предыдущей записи при условии, что i соответству-
           ющим образом описано раньше.
           
                1.4.9 Описания
           
             Описание - это оператор, вводящий имя в программе. Оно может так-
           же инициализировать объект с этим именем. Выполнение описания озна-
           чает,  что когда поток управления доходит до описания,  вычисляется
           инициализирующее выражение (инициализатор) и производится инициали-
           зация. Например:
           
             for (int i = 1; i
             //...
             x = sqrt(4);
           
             Поскольку обычные заголовочные файлы включаются во многие  исход-
           ные файлы, они не содержат описаний, которые не должны повторяться.
           Например, тела функций даются только для inline-подставляемых функ-
           ций  (#1.12)  и инициализаторы даются только для констант (#1.3.1).
           За исключением этих случаев,  заголовочный файл является хранилищем
           информации о типах. Он обеспечивает интерфейс между отдельно компи-
           лируемыми частями программы.
           
             В команде включения include  имя  файла,  заключенное  в  угловые
           скобки,  например , относится к файлу с этим именем в стан-
           дартном каталоге (часто это /usr/include/CC), на файлы, находящиеся
           в каких-либо других местах ссылаются с помощью имен,  заключенных в
           двойные кавычки. Например:
           
             #include "math1.h"
             #include "/usr/bs/math2.h"
           
           включит math1.h  из текущего пользовательского каталога,  а math2.h
           из каталога /usr/bs.
           
             Здесь приводится очень маленький законченный пример программы,  в
           котором строка определяется в одном файле, а ее печать производится
           в другом. Файл header.h определяет необходимые типы:
           
             // header.h
           
             extern char* prog_name; extern void f(); В файле main.c находится
             главная программа:
           
             // main.c
           
             #include "header.h"
             char* prog_name = "дурацкий, но полный"; main() {
                 f();
             }
           
           а файл f.c печатает строку:
           
             // f.c
           
             #include 
             #include "header.h"
             void f() {
                 cout << prog_name << "\n";
             }
           
             Скомпилировать и запустить программу вы можете например так:
           
             $ CC main.c f.c -o silly
             $ silly
             дурацкий, но полный
             $
           
                1.7 Классы
           
             Давайте посмотрим,  как  мы могли бы определить тип потока вывода
           ostream.  Чтобы упростить задачу,  предположим, что для буферизации
           определен  тип  streambuf.  Тип streambuf на самом деле определен в
           , где также находится и настоящее определение ostream.
           
             Пожалуйста, не испытывайте примеры, определяющие ostream в этом и
           последующих разделах. Пока вы не сможете полностью избежать исполь-
           зования , компилятор будет возражать против переопределе-
           ний.
           
             Определение типа,  определяемого пользователем (который в С++ на-
           зывается class,  т.е. класс), специфицирует данные, необходимые для
           представления объекта этого типа, и множество операций для работы с
           этими объектами.  Определение имеет две части:  закрытую  (private)
           часть, содержащую информацию, которой может пользоваться только его
           разработчик,  и открытую (public) часть,  представляющую  интерфейс
           типа с пользователем:
           
             class ostream { streambuf* buf; int state;
             public:
                 void put(char*); void put(long); void put(double);
             }
             Описания после метки public: задают интерфейс: пользователь может
           обращаться только к трем  функциям  put().  Описания  перед  меткой
           public  задают  представление  объекта класса ostream.  Имена buf и
           state могут использоваться только функциями put(), описанными в от-
           крытой части.
           
             class определяет тип, а не объект данных, поэтому чтобы использо-
           вать ostream,  мы должны один такой объект описать (так же,  как мы
           описываем переменные типа int):
           
             ostream my_out;
           
             Считая, что my_out был соответствующим образом проинициализирован
           (как, объясняется в #1.10), его можно использовать например так:
           
             my_out.put("Hello, world\n");
           
             С помощью операции точка выбирается член класса для данного  объ-
           екта этого класса. Здесь для объекта my_out вызывается член функция
           put().
           
             Функция может определяться так:
           
             void ostream::put(char* p) {
                 while (*p) buf.sputc(*p++);
             }
           
           где sputc() - функция, которая помещает символ в streambuf. Префикс
           ostream необходим, чтобы отличить put() ostream'а от других функций
           с именем put().
           
             Для обращения к функции члену должен быть указан объект класса. В
           функции члене можно ссылаться на этот объект неявно,  как это дела-
           лось выше в ostream::put():  в каждом вызове buf относится к  члену
           buf объекта, для которого функция вызвана.
           
             Можно также ссылаться на этот объект явно посредством указателя с
           именем this.  В функции члене класса X this неявно  описан  как  X*
           (указатель  на  X) и инициализирован указателем на тот объект,  для
           которого эта функция вызвана. Определение ostream::put() можно так-
           же записать в виде:
           
             void ostream::put(char* p) {
                 while (*p) this->buf.sputc(*p++);
             }
           
             Операция -> применяется для выбора члена объекта,  заданного ука-
           зателем.
                1.8 Перегрузка операций
           
             Настоящий класс  ostream  определяет  операцию <<,  чтобы сделать
           удобным вывод нескольких объектов одним оператором. Давайте посмот-
           рим, как это сделано.
           
             Чтобы определить  @,  где  @ - некоторая операция языка С++,  для
           каждого определяемого пользователем типа вы определяете  функцию  с
           именем operator@, которая получает параметры соответствующего типа.
           Например:
           
             class ostream {
                 //...
                 ostream operator<<(char*);
             };
           
             ostream ostream::operator<<(char* p) {
                 while (*p) buf.sputc(*p++); return *this;
             }
           
           определяет операцию  <<  как член класса ostream,  поэтому s<

"); а если применить операцию взятия адреса, то вы получите адрес объ- екта, на который ссылается ссылка: &s1 == &my_out Первая очевидная польза от ссылок состоит в том, чтобы обеспечить передачу адреса объекта, а не самого объекта, в функцию вывода (в некоторых языках это называется вызов по ссылке): ostream& operator<<(ostream& s, complex z) { return s << "(" << z.real << "," << z.imag << ")"; } Достаточно интересно, что тело функции осталось без изменений, но если вы будете осуществлять присваивание s, то будете воздейство- вать на сам объект, а не на его копию. В данном случае то, что возвращается ссылка, также повышает эффективность, поскольку оче- видный способ реализации ссылки - это указатель, а передача указа- теля гораздо дешевле, чем передача большой структуры данных. Ссылки также существенны для определения потока ввода, поскольку операция ввода получает в качестве операнда переменную для считыва- ния. Если бы ссылки не использовались, то пользователь должен был бы явно передавать указатели в функции ввода. class istream { //... int state; public: istream& operator>>(char&); istream& operator>>(char*); istream& operator>>(int&); istream& operator>>(long&); //... }; Заметьте, что для чтения long и int используются разные функции, тогда как для их печати требовалась только одна. Это вполне обычно, и причина в том, что int может быть преобразовано в long по стан- дартным правилам неявного преобразования (#с.6.6), избавляя таким образом программиста от беспокойства по поводу написания обеих функций ввода. 1.10 Конструкторы Определение ostream как класса сделало члены данные закрытыми. Только функция член имеет доступ к закрытым членам, поэтому надо предусмотреть функцию для инициализации. Такая функция называется конструктором и отличается тем, что имеет то же имя, что и ее класс: class ostream { //... ostream(streambuf*); ostream(int size, char* s); }; Здесь задано два конструктора. Один получает вышеупомянутый streambuf для реального вывода, другой получает размер и указатель на символ для форматирования строки. В описании необходимый для конструктора список параметров присоединяется к имени. Теперь вы можете, например, описать такие потоки: ostream my_out(&some_stream_buffer); char xx[256]; ostream xx_stream(256,xx); Описание my_out не только задает соответствующий объем памяти где-то в другом месте, оно также вызывает конструктор ostream::ostream(streambuf*), чтобы инициализировать его параметром &some_stream_buffer, предположительно указателем на подходящий объ- ект класса streambuf. Описание конструкторов для класса не только дает способ инициализации объектов, но также обеспечивает то, что все объекты этого класса будут проинициализированы. Если для класса были описаны конструкторы, то невозможно описать переменную этого класса так, чтобы конструктор не был вызван. Если класс имеет конструктор, не получающий параметров, то этот конструктор будет вызываться в том случае, если в описании нет ни одного параметра. 1.11 Вектора Встроенное в С++ понятие вектора было разработано так, чтобы обеспечить максимальную эффективность выполнения при минимальном расходе памяти. Оно также (особенно когда используется совместно с указателями) является весьма универсальным инструментом для постро- ения средств более высокого уровня. Вы могли бы, конечно, возра- зить, что размер вектора должен задаваться как константа, что нет проверки выхода за границы вектора и т.д. Ответ на подобные возра- жения таков: "Вы можете запрограммировать это сами." Давайте посмотрим, действительно ли оправдан такой ответ. Другими словами, проверим средства абстракции языка С++, попытавшись реализовать эти возможности для векторных типов, которые мы создадим сами, и посмотрим, какие с этим связаны трудности, каких это требует зат- рат, и насколько получившиеся векторные типы удобны в обращении. class vector { int* v; int sz; public: vector(int); // конструктор ~vector(); // деструктор int size() { return sz; } void set_size(int); int& operator[](int); int& elem(int i) { return v[i]; } }; Функция size возвращает число элементов вектора, таким образом индексы должны лежать в диапазоне 0 ... size()-1. Функция set_size сделана для изменения этого размера, elem обеспечивает доступ к элементам без проверки индекса, а operator[] дает доступ с провер- кой границ. Идея состоит в том, чтобы класс сам был структурой фиксированного размера, управляющей доступом к фактической памяти вектора, которая выделяется конструктором вектора с помощью распределителя свободной памяти new: vector::vector(int s) { if (s<=0) error("bad vector size"); // плохой размер вектора sz = s; v = new int[s]; } Теперь вы можете описывать вектора типа vector почти столь же элегантно, как и вектора, встроенные в сам язык: vector v1(100); vector v2(nelem*2-4); Операцию доступа можно определить как int& vector::operator[](int i) { if(i<0 || sz<=i) error("vector index out of range"); // индекс выходит за границы вектора return v[i]; } Операция || (ИЛИИЛИ) - это логическая операция ИЛИ. Ее правый операнд вычисляется только тогда, когда это необходимо, то есть если вычисление левого операнда дало ноль. Возвращение ссылки обеспечивает то, что запись [] может использоваться с любой стороны операции присваивания: v1[x] = v2[y]; Функция со странным именем ~vector - это деструктор, то есть функция, описанная для того, чтобы она неявно вызывалась, когда объект класса выходит из области видимости. Деструктор класса C имеет имя ~C. Если его определить как vector::~vector() { delete v; } то он будет, с помощью операции delete, освобождать пространство, выделенное конструктором, поэтому когда vector выходит из области видимости, все его пространство возвращается обратно в память для дальнейшего использования. 1.12 Inline-подстановка Если часто повторяется обращение к очень маленькой функции, то вы можете начать беспокоиться о стоимости вызова функции. Обращение к функции члену не дороже обращения к функции не члену с тем же числом параметров (надо помнить, что функция член всегда имеет хотя бы один параметр), и вызовы в функций в С++ примерно столь же эф- фективны, сколь и в любом языке. Однако для слишком маленьких функ- ций может встать вопрос о накладных расходах на обращение. В этом случае можно рассмотреть возможность спецификации функции как inline-подставляемой. Если вы поступите таким образом, то компиля- тор сгенерирует для функции соответствующий код в месте ее вызова. Семантика вызова не изменяется. Если, например, size и elem inline-подставляемые, то vector s(100); //... i = s.size(); x = elem(i-1); порождает код, эквивалентный //... i = 100; x = s.v[i-1]; С++ компилятор обычно достаточно разумен, чтобы генерировать настолько хороший код, насколько вы можете получить в результате прямого макрорасширения. Разумеется, компилятор иногда вынужден использовать временные переменные и другие уловки, чтобы сохранить семантику. Вы можете указать, что вы хотите, чтобы функция была inline-подставляемой, поставив ключевое слово inline, или, для функции члена, просто включив определение функции в описание класса, как это сделано в предыдущем примере для size() и elem(). При хорошем использовании inline-функции резко повышают скорость выполнения и уменьшают размер объектного кода. Однако, inline- функции запутывают описания и могут замедлить компиляцию, поэтому, если они не необходимы, то их желательно избегать. Чтобы inline- функция давала существенный выигрыш по сравнению с обычной функци- ей, она должна быть очень маленькой. 1.13 Производные классы Теперь давайте определим вектор, для которого пользователь может задавать границы изменения индекса. class vec: public vector { int low, high; public: vec(int,int); int& elem(int); int& operator[](int); }; Определение vec как :public vector означает, в первую очередь, что vec - это vector. То есть, тип vec имеет (наследует) все свойства типа vector дополнительно к тем, что описаны специально для него. Говорят, что vector является базовым классом для vec, а о vec говорится, что он производный класс от vector. Класс vec модифицирует класс vector тем, что в нем задается дру- гой конструктор, который требует от пользователя указывать две гра- ницы изменения индекса, а не длину, и имеются свои собственные функции доступа elem(int) и operator[](int). Функция elem() класса vec легко выражается через elem() класса vector: int& vec::elem(int i) { return vector::elem(i-low); } Операция разрешения области видимости :: используется для того, чтобы не было бесконечной рекурсии обращения к vec::elem() из нее самой. с помощью унарной операции :: можно ссылаться на нелокальные имена. Было бы разумно описать vec::elem() как inline, поскольку, скорее всего, эффективность существенна, но необязательно, неразум- но и невозможно написать ее так, чтобы она непосредственно исполь- зовала закрытый член v класса vector. Функции производного класса не имеют специального доступа к закрытым членам его базового класса. Конструктор можно написать так: vec::vec(int lb, int hb) : (hb-lb+1) { if (hb-lb<0) hb = lb; low = lb; high = hb; } Запись: (hb-lb+1) используется для определения списка параметров конструктора базового класса vector::vector(). Этот конструктор вы- зывается перед телом vec::vec(). Вот небольшой пример, который мож- но запустить, если скомпилировать его вместе с остальными описания- ми vector: #include void error(char* p) { cerr << p << "\n"; // cerr - выходной поток сообщений об ошибках exit(1); } void vector::set_size(int) { /* пустышка */ } int& vec::operator[](int i) { if (i void error(char* p) { cerr << p << "\n"; exit(1); } void vector::set_size(int) { /*...*/ } int& vec::operator[](int i) { /*...*/ } main() { Vec a(10); Vec b(10); for (int i=0; i declare(vector,int); main() { vector(int) vv(10); vv[2] = 3; vv[10] = 4; // ошибка: выход за границы } Файл vector.h таким образом определяет макросы, чтобы макрос declare(vector,int) после расширения превращался в описание класса vector, очень похожий на тот, который был определен выше, а макрос implement(vector,int) расширялся в определение функций этого класса. Поскольку макрос implement(vector,int) в результате расши- рения превращается в определение функций, его можно использовать в программе только один раз, в то время как declare(vector,int) долж- но использоваться по одному разу в каждом файле, работающем с этим типом целых векторов. declare(vector,char); //... implement(vector,char); даст вам отдельный тип "вектор символов". Пример реализации обоб- щенных классов с помощью макросов приведен в #7.3.5. 1.17 Полиморфные Вектора У вас есть другая возможность - определить ваш векторный и другие вмещающие классы через указатели на объекты некоторого класса: class common { //... }; class vector { common** v; //... public: cvector(int); common*& elem(int); common*& operator[](int); //... }; Заметьте, что поскольку в таких векторах хранятся указатели, а не сами объекты, объект может быть "в" нескольких таких векторах од- новременно. Это очень полезное свойство подобных вмещающих классов, таких, как вектора, связанные списки, множества и т.д. Кроме того, можно присваивать указатель на производный класс указателю на его базовый класс, поэтому можно использовать приведенный выше cvector для хранения указателей на объекты всех производных от common классов. Например: class apple : public common { /*...*/ } class orange : public common { /*...*/ } class apple_vector : public cvector { public: cvector fruitbowl(100); //... apple aa; orange oo; //... fruitbowl[0] = &aa; fruitbowl[1] = &oo; } Однако, точный тип объекта, вошедшего в такой вмещающий класс, больше компилятору не известен. Например, в предыдущем примере вы знаете, что элемент вектора является common, но является он apple или orange? Обычно точный тип должен впоследствии быть восстанов- лен, чтобы обеспечить правильное использование объекта. Для этого нужно или в какой-то форме хранить информацию о типе в самом объек- те, или обеспечить, чтобы во вмещающий класс помещались только объ- екты данного типа. Последнее легко достигается с помощью производ- ного класса. Вы можете, например, создать вектор указателей на apple: class apple_vector : public cvector { public: apple*& elem(int i) { return (apple*&) cvector::elem(i); } //... }; используя запись приведения к типу (тип)выражение, чтобы преобразо- вать common*& (ссылку на указатель на common), которую возвращает cvector::elem, в apple*&. Такое применение производных классов соз- дает альтернативу обобщенным классам. Писать его немного труднее (если не использовать макросы таким образом, чтобы производные классы фактически реализовывали обобщенные классы, см. #7.3.5), но оно имеет то преимущество, что все производные классы совместно используют единственную копию функции базового класса. В случае обобщенных классов, таких, как vector(type), для каждого нового используемого типа должна создаваться (с помощью implement()) новая копия таких функций. Другой способ, хранение идентификации типа в каждом объекте, приводит нас к стилю программирования, который часто называют объекто-основанным или объектно-ориентированным. 1.18 Виртуальные Функции Предположим, что мы пишем программу для изображения фигур на эк- ране. Общие атрибуты фигуры представлены классом shape, а специаль- ные атрибуты - специальными классами: class shape { point center; color col; //... public: void move(point to) { center=to; draw(); } point where() { return center; } virtual void draw(); virtual void rotate(int); //... }; Функции, которые можно определить не зная точно определенной фи- гуры (например, move и where, то есть, "передвинуть" и "где"), мож- но описать как обычно. Остальные функции описываются как virtual, то есть такие, которые должны определяться в производном классе. Например: class circle: public shape { int radius; public: void draw(); void rotatte(int i) {} //... }; Теперь, если shape_vec - вектор фигур, то можно написать: for (int i = 0; i



Яндекс цитирования