Главная » Просмотр файлов » Бьерн Страуструп

Бьерн Страуструп (947334), страница 74

Файл №947334 Бьерн Страуструп (Стpаустpуп - Книга о C++) 74 страницаБьерн Страуструп (947334) страница 742013-09-15СтудИзба
Просмтор этого файла доступен только зарегистрированным пользователям. Но у нас супер быстрая регистрация: достаточно только электронной почты!

Текст из файла (страница 74)

"программируемые" отношения трудно выразить на языках программирования,

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

Например, такое средство может не отличить "делегирование" от B

к A с помощью A::p от любого другого использования B*.

Все-таки следует всюду, где это возможно, добиваться

взаимнооднозначного соответствия между понятиями проекта и понятиями

языка программирования. Оно дает определенную простоту и гарантирует,

что проект адекватно отображается в программе, что упрощает

работу программиста и вспомогательных средств.

Операции преобразований типа являются механизмом, с помощью которого

можно представить в языке класс программируемых отношений, а именно:

операция преобразования X::operator Y() гарантирует, что всюду,

где допустимо использование Y, можно применять и X. Такое же

отношение задает конструктор Y::Y(X). Отметим, что операция

преобразования типа (как и конструктор) скорее создает новый объект,

чем изменяет тип существующего объекта. Задать операцию преобразования

к функции Y - означает просто потребовать неявного применения

функции, возвращающей Y. Поскольку неявные применения операций

преобразования типа и операций, определяемых конструкторами, могут

привести к неприятностям, полезно проанализировать их в отдельности

еще в проекте.

Важно убедиться, что граф применений операций преобразования типа

не содержит циклов. Если они есть, возникает двусмысленная ситуация,

при которой типы, участвующие в циклах, становятся несовместимыми в

комбинации. Например:

class Big_int {

//...

friend Big_int operator+(Big_int,Big_int);

//...

operator Rational();

//...

};

class Rational {

//...

friend Rational operator+(Rational,Rational);

//...

operator Big_int();

};

Типы Rational и Big_int не так гладко взаимодействуют, как можно

было бы подумать:

void f(Rational r, Big_int i)

{

//...

g(r+i); // ошибка, неоднозначность:

// operator+(r,Rational(i)) или

// operator+(Big_int(r),i)

g(r,Rational(i)); // явное разрешение неопределенности

g(Big_int(r),i); // еще одно

}

Можно было бы избежать таких "взаимных" преобразований, сделав

некоторые из них явными. Например, преобразование Big_int к типу

Rational можно было бы задать явно с помощью функции make_Rational()

вместо операции преобразования, тогда сложение в приведенном

примере разрешалось бы как g(BIg_int(r),i). Если нельзя избежать

"взаимных" операций преобразования типов, то нужно преодолевать

возникающие столкновения или с помощью явных преобразований (как было

показано), или с помощью определения нескольких различных версий

бинарной операции (в нашем случае +).

12.3 Компоненты

В языке С++ нет конструкций, которые могут выразить прямо в программе

понятие компонента, т.е. множества связанных классов. Основная

причина этого в том, что множество классов (возможно с соответствующими

глобальными функциями и т.п.) может соединяться в компонент по

самым разным признакам. Отсутствие явного представления понятия в

языке затрудняет проведение границы между информацией (имена),

используемой внутри компонента, и информацией (имена), передаваемой

из компонента пользователям.

В идеале, компонент определяется множеством интерфейсов, используемых

для его реализации, плюс множеством интерфейсов, представляемых

пользователем, а все прочее считается "спецификой реализации" и

должно быть скрыто от остальных частей системы. Таково может быть

в действительности представление о компоненте у разработчика.

Программист должен смириться с тем фактом, что С++ не дает

общего понятия пространства имен компонента, так что его

приходится "моделировать" с помощью понятий классов и единиц

трансляции, т.е. тех средств, которые есть в С++ для ограничения

области действия нелокальных имен.

Рассмотрим два класса, которые должны совместно использовать

функцию f() и переменную v. Проще всего описать f и v как

глобальные имена. Однако, всякий опытный программист знает, что

такое "засорение" пространства имен может привести в конце концов

к неприятностям: кто-то может ненарочно использовать имена f или v

не по назначению или нарочно обратиться к f или v,

прямо используя "специфику реализации" и обойдя тем самым явный

интерфейс компонента. Здесь возможны три решения:

[1] Дать "необычные" имена объектам и функциям, которые не

рассчитаны на пользователя.

[2] Объекты или функции, не предназначенные для пользователя,

описать в одном из файлов программы как статические (static).

[3] Поместить объекты и функции, не предназначенные для пользователя,

в класс, определение которого закрыто для пользователей.

Первое решение примитивно и достаточно неудобно для создателя

программы, но оно действует:

// не используйте специфику реализации compX,

// если только вы не разработчик compX:

extern void compX_f(T2*, const char*);

extern T3 compX_v;

// ...

Такие имена как compX_f и compX_v вряд ли могут привести к коллизии, а на

тот довод, что пользователь может быть злоумышленником и использовать

эти имена прямо, можно ответить, что пользователь в любом случае может

оказаться злоумышленником, и что языковые механизмы защиты предохраняют

от несчастного случая, а не от злого умысла. Преимущество этого

решения в том, что оно применимо всегда и хорошо известно. В то же

время оно некрасиво, ненадежно и усложняет ввод текста.

Второе решение более надежно, но менее универсально:

// специфика реализации compX:

static void compX_f(T2* a1, const char *a2) { /* ... */ }

static T3 compX_v;

// ...

Трудно гарантировать, что информация, используемая в классах одного

компонента, будет доступна только в одной единице трансляции,

поскольку операции, работающие с этой информацией, должны

быть доступны везде. Это решение может к тому же привести к

громадным единицам трансляции, а в некоторых отладчиках для С++

не организован доступ к именам статических функций и переменных.

В то же время это решение надежно и часто оптимально для небольших

компонентов.

Третье решение можно рассматривать как формализацию и обобщение

первых двух:

class compX_details { // специфика реализации compX

public:

static void f(T2*, const char*);

static T3 v;

// ...

};

Описание compX_details будет использовать только создатель класса,

остальные не должны включать его в свои программы.

В компоненте конечно может быть много классов, не предназначенных

для общего пользования. Если их имена тоже рассчитаны только на

локальное использование, то их также можно "спрятать" внутри

классов, содержащих специфику реализации:

class compX_details { // специфика реализации compX.

public:

// ...

class widget {

// ...

};

// ...

};

Укажем, что вложенность создает барьер для использования widget

в других частях программы. Обычно классы, представляющие

ясные понятия, считаются первыми кандидатами на повторное

использование, и, значит составляют часть интерфейса компонента,

а не деталь реализации. Другими словами, хотя для сохранения

надлежащего уровня абстракции вложенные объекты, используемые для

представления некоторого объекта класса, лучше считать скрытыми

деталями реализации, классы, определяющие такие вложенные объекты,

лучше не делать скрытыми, если они имеют достаточную общность.

Так, в следующем примере упрятывание, пожалуй, излишне:

class Car {

class Wheel {

// ...

};

Wheel flw, frw, rlw, rrw;

// ...

};

Во многих ситуациях для поддержания уровня абстракции понятия

машины (Car) следует упрятывать реальные колеса (класс Wheel),

ведь когда вы работаете с машиной, вы не можете независимо от нее

использовать колеса. С другой стороны, сам класс Wheel является

вполне подходящим для широкого использования, поэтому лучше

вынести его определение из класса Car:

class Wheel {

// ...

};

class Car {

Wheel flw, frw, rlw, rrw;

// ...

};

Использовать ли вложенность? Ответ на этот вопрос зависит

от целей проекта и общности используемых понятий. Как вложенность,

так и ее отсутствие могут быть вполне допустимыми решениями для данного

проекта. Но поскольку вложенность предохраняет от засорения

общего пространства имен, в своде правил ниже рекомендуется

использовать вложенность, если только нет причин не делать этого.

Отметим, что заголовочные файлы дают мощное средство для

различных представлений компонент разным пользователям, и они же

позволяют удалять из представления компонента для пользователя те

классы, которые связаны со спецификой реализации.

Другим средством построения компонента и представления его

пользователю служит иерархия. Тогда базовый класс используется как

хранилище общих данных и функций. Таким способом устраняется

проблема, связанная с глобальными данными и функциями, предназначенными

для реализации общих запросов классов данного компонента.

С другой стороны, при таком решении классы компонента становятся

слишком связанными друг с другом, а пользователь попадает в зависимость

от всех базовых классов тех компонентов, которые ему действительно

нужны. Здесь также проявляется тенденция к тому, что члены,

представляющие "полезные" функции и данные "всплывают" к базовому

классу, так что при слишком большой иерархии классов проблемы с

глобальными данными и функциями проявятся уже в рамках этой иерархии.

Вероятнее всего, это произойдет для иерархии с одним корнем, а для

борьбы с этим явлением можно применять виртуальные базовые классы

($$6.5.4). Иногда лучше выбрать иерархию для представления компонента,

а иногда нет. Как всегда сделать выбор предстоит разработчику.

12.4 Интерфейсы и реализации

Идеальный интерфейс должен

- представлять полное и согласованное множество понятий для

пользователя,

- быть согласованным для всех частей компонента,

- скрывать специфику реализации от пользователя,

- допускать несколько реализаций,

- иметь статическую систему типов,

- определяться с помощью типов из области приложения,

- зависеть от других интерфейсов лишь частично и вполне определенным

образом.

Отметив необходимость согласованности для всех классов, которые

образуют интерфейс компонента с остальным миром, мы можем упростить

вопрос интерфейса, рассмотрев только один класс, например:

class X { // пример плохого определения интерфейса

Y a;

Z b;

public:

void f(const char* ...);

void g(int[],int);

void set_a(Y&);

Y& get_a();

};

В этом интерфейсе содержится ряд потенциальных проблем:

-Типы Y и Z используются так, что определения Y и Z должны быть

известны во время трансляции.

- У функции X::f может быть произвольное число параметров

неизвестного типа (возможно, они каким-то образом контролируются

"строкой формата", которая передается в качестве первого

параметра).

- Функция X::g имеет параметр типа int[]. Возможно это нормально,

но обычно это свидетельствует о том, что определение слишком

низкого уровня абстракции. Массив целых не является достаточным

определением, так как неизвестно из скольких он может

состоять элементов.

- Функции set_a() и get_a(), по всей видимости, раскрывают

представление объектов класса X, разрешая прямой доступ

к X::a.

Здесь функции-члены образуют интерфейс на слишком низком уровне

абстракции. Как правило классы с интерфейсом такого уровня относятся

к специфике реализации большого компонента, если они вообще могут

к чему-нибудь относиться. В идеале параметр функции из интерфейса

должен сопровождаться такой информацией, которой достаточно

для его понимания. Можно сформулировать такое правило: надо уметь

передавать запросы на обслуживание удаленному серверу по узкому

каналу.

Язык С++ раскрывает представление класса как часть интерфейса.

Это представление может быть скрытым (с помощью private или

protected), но обязательно доступным транслятору, чтобы он мог разместить

автоматические (локальные) переменные, сделать подстановку тела

функции и т.д. Отрицательным следствием этого является то, что

Характеристики

Тип файла
Документ
Размер
4,26 Mb
Тип материала
Учебное заведение
Неизвестно

Список файлов книги

Свежие статьи
Популярно сейчас
Зачем заказывать выполнение своего задания, если оно уже было выполнено много много раз? Его можно просто купить или даже скачать бесплатно на СтудИзбе. Найдите нужный учебный материал у нас!
Ответы на популярные вопросы
Да! Наши авторы собирают и выкладывают те работы, которые сдаются в Вашем учебном заведении ежегодно и уже проверены преподавателями.
Да! У нас любой человек может выложить любую учебную работу и зарабатывать на её продажах! Но каждый учебный материал публикуется только после тщательной проверки администрацией.
Вернём деньги! А если быть более точными, то автору даётся немного времени на исправление, а если не исправит или выйдет время, то вернём деньги в полном объёме!
Да! На равне с готовыми студенческими работами у нас продаются услуги. Цены на услуги видны сразу, то есть Вам нужно только указать параметры и сразу можно оплачивать.
Отзывы студентов
Ставлю 10/10
Все нравится, очень удобный сайт, помогает в учебе. Кроме этого, можно заработать самому, выставляя готовые учебные материалы на продажу здесь. Рейтинги и отзывы на преподавателей очень помогают сориентироваться в начале нового семестра. Спасибо за такую функцию. Ставлю максимальную оценку.
Лучшая платформа для успешной сдачи сессии
Познакомился со СтудИзбой благодаря своему другу, очень нравится интерфейс, количество доступных файлов, цена, в общем, все прекрасно. Даже сам продаю какие-то свои работы.
Студизба ван лав ❤
Очень офигенный сайт для студентов. Много полезных учебных материалов. Пользуюсь студизбой с октября 2021 года. Серьёзных нареканий нет. Хотелось бы, что бы ввели подписочную модель и сделали материалы дешевле 300 рублей в рамках подписки бесплатными.
Отличный сайт
Лично меня всё устраивает - и покупка, и продажа; и цены, и возможность предпросмотра куска файла, и обилие бесплатных файлов (в подборках по авторам, читай, ВУЗам и факультетам). Есть определённые баги, но всё решаемо, да и администраторы реагируют в течение суток.
Маленький отзыв о большом помощнике!
Студизба спасает в те моменты, когда сроки горят, а работ накопилось достаточно. Довольно удобный сайт с простой навигацией и огромным количеством материалов.
Студ. Изба как крупнейший сборник работ для студентов
Тут дофига бывает всего полезного. Печально, что бывают предметы по которым даже одного бесплатного решения нет, но это скорее вопрос к студентам. В остальном всё здорово.
Спасательный островок
Если уже не успеваешь разобраться или застрял на каком-то задание поможет тебе быстро и недорого решить твою проблему.
Всё и так отлично
Всё очень удобно. Особенно круто, что есть система бонусов и можно выводить остатки денег. Очень много качественных бесплатных файлов.
Отзыв о системе "Студизба"
Отличная платформа для распространения работ, востребованных студентами. Хорошо налаженная и качественная работа сайта, огромная база заданий и аудитория.
Отличный помощник
Отличный сайт с кучей полезных файлов, позволяющий найти много методичек / учебников / отзывов о вузах и преподователях.
Отлично помогает студентам в любой момент для решения трудных и незамедлительных задач
Хотелось бы больше конкретной информации о преподавателях. А так в принципе хороший сайт, всегда им пользуюсь и ни разу не было желания прекратить. Хороший сайт для помощи студентам, удобный и приятный интерфейс. Из недостатков можно выделить только отсутствия небольшого количества файлов.
Спасибо за шикарный сайт
Великолепный сайт на котором студент за не большие деньги может найти помощь с дз, проектами курсовыми, лабораторными, а также узнать отзывы на преподавателей и бесплатно скачать пособия.
Популярные преподаватели
Добавляйте материалы
и зарабатывайте!
Продажи идут автоматически
6309
Авторов
на СтудИзбе
313
Средний доход
с одного платного файла
Обучение Подробнее