Программирование >>  Поддержка объектно-ориентированного программирования 

1 ... 54 55 56 [ 57 ] 58 59 60 ... 120


10. (*2) Опишите класс графических объектов с набором возможных операций, который будет общим базовым в библиотеке графических объектов. Исследуйте какие-нибудь графические библиотеки, чтобы понять, какие операции нужны. Определите класс объектов базы данных с набором возможных операций, который будет общим базовым классом объектов, хранящихся как последовательность полей базы данных. Исследуйте какие-нибудь базы данных, чтобы понять, какие операции нужны. Определите объект графической базы данных, используя или не используя множественное наследование. Обсудите относительные плюсы и минусы обоих решений.

11. (*2) Напишите вариант функции clone() из $$6.7.1, в котором размножаемый объект может помещаться в область Arena ($$6.7.2), передаваемую как параметр. Реализуйте простой класс Arena как производный от Arena.

12. (*2) Пусть есть классы Circle (окружность), Square (квадрат) и Triangle (треугольник), производные от класса shape. Определите функцию intersect() с двумя параметрами типа Shape*, которая вызывает подходящую функцию, чтобы выяснить, пересекаются ли заданные две фигуры. Для этого в указанных классах нужно определить соответствующие виртуальные функции. Не тратьте силы на функцию, которая действительно устанавливает, что фигуры пересекаются, добейтесь только правильной последовательности вызовов функций.

1 3. (*5) Разработайте и реализуйте библиотеку для моделирования, управляемого событиями. Подсказка: используйте <task.h>. Там уже устаревшие функции и можно написать лучше. Должен быть класс task (задача). Объект task должен уметь сохранять свое состояние и восстанавливать его (для этого можно определить функции task::save() и task::restore()) и тогда он может действовать как сопрограмма. Специальные задачи можно определять как объекты классов, производных от task. Программу, которую выполняет задача, определите как виртуальную функцию. Должна быть возможность передавать параметры новой задаче как параметры ее конструктору или конструкторам. Должен быть диспетчер, который реализует понятие виртуального времени. Определите функцию task::delay(long), которая будет съедать виртуальное время. Важный вопрос разработки: является ли диспетчер частью класса task, или он должен быть независимым? Задачи должны иметь возможность общения друг с другом. Для этой цели разработайте класс queue (очередь). Придумайте способ, чтобы задача могла ожидать входной поток из нескольких очередей. Все динамические ошибки должны обрабатываться единообразно. Как организовать отладку программ, написанных с помощью такой библиотеки?



ГЛАВА 7.

Если я выбираю слово, оно значит только то, что я решу, ни больше и ни меньше.

- Шалтай Болтай

Глава содержит описание механизма перегрузки операций в С++. Программист может задать интерпретацию операций, когда они применяются к объектам определенного класса. Помимо арифметических, логических и операций отношения можно переопределить вызов функций (), индексацию [], косвенное обращение ->, а также присваивание и инициализацию. Можно определить явные и скрытые преобразования между пользовательскими и основными типами. Показано, как определить класс, объект которого можно копировать и уничтожать только с помощью специальных, определенных пользователем функций.

7.1 Введение

Обычно в программах используются объекты, являющиеся конкретным представлением абстрактных понятий. Например, в С++ тип данных int вместе с операциями +, -, *, / и т.д. реализует (хотя и ограниченно) математическое понятие целого. Обычно с понятием связывается набор действий, которые реализуются в языке в виде основных операций над объектами, задаваемых в сжатом, удобном и привычном виде. К сожалению, в языках программирования непосредственно представляется только малое число понятий. Так, понятия комплексных чисел, алгебры матриц, логических сигналов и строк в С++ не имеют непосредственного выражения. Возможность задать представление сложных объектов вместе с набором операций, выполняемых над такими объектами, реализуют в С++ классы. Позволяя программисту определять операции над объектами классов, мы получаем более удобную и традиционную систему обозначений для работы с этими объектами по сравнению с той, в которой все операции задаются как обычные функции. Приведем пример:

class complex { double re, im; public:

complex(double r, double i) { re=r; im=i; } friend complex operator+(complex, complex); friend complex operator*(complex, complex);

Здесь приведена простая реализация понятия комплексного числа, когда оно представлено парой чисел с плавающей точкой двойной точности, с которыми можно оперировать только с помощью операций + и *. Интерпретацию этих операций задает программист в определениях функций с именами operator+ и operator*. Так, если b и c имеют тип complex, то b+c означает (по определению) operator+(b,c). Теперь можно приблизиться к привычной записи комплексных выражений:

void f()

complex a = complex(1,3.1); complex b = complex(1.2,2); complex c = b; a = b+c; b = b+c*a;

c = a*b+complex(1,2);

Сохраняются обычные приоритеты операций, поэтому второе выражение выполняется как b=b+(c*a), а не как b=(b+c)*a.

7.2 Операторные функции

Можно описать функции, определяющие интерпретацию следующих операций:



&

<

>

&=

<<

>>

<<=

<=

> =

&&

->*

->

delete

Последние пять операций означают: косвенное обращение ($$7.9), индексацию ($$7.7), вызов функции ($$7.8), размещение в свободной памяти и освобождение ($$3.2.6). Нельзя изменить приоритеты этих операций, равно как и синтаксические правила для выражений. Так, нельзя определить унарную операцию % , также как и бинарную операцию !. Нельзя ввести новые лексемы для обозначения операций, но если набор операций вас не устраивает, можно воспользоваться привычным обозначением вызова функции. Поэтому используйте pow(), а не ** . Эти ограничения можно счесть драконовскими, но более свободные правила легко приводят к неоднозначности. Допустим, мы определим операцию ** как возведение в степень, что на первый взгляд кажется очевидной и простой задачей. Но если как следует подумать, то возникают вопросы: должны ли операции ** выполняться слева направо (как в Фортране) или справа налево (как в Алголе)? Как интерпретировать выражение a**p как a*(*p) или как (a)**(p)?

Именем операторной функции является служебное слово operator, за которым идет сама операция, например, operator<<. Операторная функция описывается и вызывается как обычная функция. Использование символа операции является просто краткой формой записи вызова операторной функции:

void f(complex a, complex b)

complex c = a + b; краткая форма

complex d = operator+(a,b); явный вызов

С учетом приведенного описания типа complex инициализаторы в этом примере являются эквивалентными.

7.2.1 Бинарные и унарные операции

Бинарную операцию можно определить как функцию-член с одним параметром, или как глобальную функцию с двумя параметрами. Значит, для любой бинарной операции @ выражение aa @ bb интерпретируется либо как aa.operator(bb), либо как operator@(aa,bb). Если определены обе функции, то выбор интерпретации происходит по правилам сопоставления параметров ($$R.13.2). Префиксная или постфиксная унарная операция может определяться как функция-член без параметров, или как глобальная функция с одними параметром. Для любой префиксной унарной операции @ выражение @aa интерпретируется либо как aa.operator@(), либо как operator@(aa). Если определены обе функции, то выбор интерпретации происходит по правилам сопоставления параметров ($$R.13.2). Для любой постфиксной унарной операции @ выражение @aa интерпретируется либо как aa.operator@(int), либо как operator@(aa,int). Подробно это объясняется в $$7.1 0. Если определены обе функции, то выбор интерпретации происходит по правилам сопоставления параметров ($$1 3.2). Операцию можно определить только в соответствии с синтаксическими правилами, имеющимися для нее в грамматике С++. В частности, нельзя определить % как унарную операцию, а + как тернарную. Проиллюстрируем сказанное примерами:

class X {

члены ( неявно используется указатель this):

X* operator&(); префиксная унарная операция &

( взятие адреса)

X operator&(X); бинарная операция & (И поразрядное)

X operator++(int); постфиксный инкремент

X operator&(X,X); ошибка: & не может быть тернарной

X operator/(); ошибка: / не может быть унарной

глобальные функции ( обычно друзья)

X operator-(X); префиксный унарный минус

1 82



1 ... 54 55 56 [ 57 ] 58 59 60 ... 120

© 2006 - 2024 pmbk.ru. Генерация страницы: 0
При копировании материалов приветствуются ссылки.
Яндекс.Метрика