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

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

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

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

контроля errno, и в результате возникают наведенные ошибки,

вызванные тем, что стандартные функции возвращают не то значение.

Кроме того, если в программе есть параллельные вычисления,

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

ошибках неизбежно приведет к катастрофе.

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

на которые рассчитан вариант [4] ( "вызвать функцию реакции на

ошибку"). Отметим, однако, что если особые ситуации не предусмотрены,

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

только один из трех перечисленных вариантов. Обсуждение функций

реакций и особых ситуацией будет продолжено в $$9.4.3.

Механизм особых ситуаций успешно заменяет традиционные

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

неполным, некрасивым или чреватым ошибками решением. Этот механизм

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

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

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

Свойственный этому механизму регулярный способ обработки ошибок

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

программы.

В этом способе обработки ошибок есть для программирующих на С

новый момент: стандартная реакция на ошибку (особенно на ошибку

в библиотечной функции) состоит в завершении программы. Традиционной

была реакция продолжать программу в надежде, что она как-то

завершится сама. Поэтому способ, базирующийся на особых ситуациях,

делает программу более "хрупкой" в том смысле, что требуется

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

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

развития программы (или получать их еще позже, когда программу

сочтут завершенной и передадут ничего не подозревающему пользователю).

Если завершение программы является неприемлемой реакцией, можно

смоделировать традиционную реакцию с помощью перехвата всех особых

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

классу ($$9.3.2).

Механизм особых ситуаций можно рассматривать как динамический

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

на стадии трансляции. При таком подходе более важной становится

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

процесса выполнения программы, чем для программ на С. Однако,

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

встроить в программную систему, она будет понятнее другим программистам и

с ней проще будет работать различным сервисным программам. Можно

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

подобно другим средствам С++, "хороший" стиль программирования,

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

объеме и на неформальном уровне.

Все же надо сознавать, что обработка ошибок остается трудной

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

чем традиционные способы, он все равно недостаточно структурирован

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

управления.

9.1.2 Другие точки зрения на особые ситуации

"Особая ситуация" - одно из тех понятий, которые имеют разный смысл

для разных людей. В С++ механизм особых ситуаций предназначен для

обработки ошибок. В частности, он предназначен для обработки ошибок

в программах, состоящих из независимо создаваемых компонентов.

Этот механизм рассчитан на особые ситуации, возникающие только при

последовательном выполнении программы (например, контроль границ

массива). Асинхронные особые ситуации такие, например, как прерывания

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

механизма. В различных системах существуют другие механизмы,

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

от конкретной системы.

Механизм особых ситуаций является конструкцией с нелокальной

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

return. Поэтому особые ситуации можно использовать для целей, никак

не связанных с обработкой ошибок ($$9.5). Все-таки основным

назначением механизма особых ситуаций и темой этой главы будет

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

9.2 Различение особых ситуаций

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

ошибок. Эти ошибки можно сопоставить с особыми ситуациями, имеющими

различные имена. Так, в классе Vector обычно приходится выявлять

и сообщать об ошибках двух видов: ошибки диапазона и ошибки,

вызванные неподходящим для конструктора параметром:

class Vector {

int* p;

int sz;

public:

enum { max = 32000 };

class Range { }; // особая ситуация индекса

class Size { }; // особая ситуация "неверный размер"

Vector(int sz);

int& operator[](int i);

// ...

};

Как было сказано, операция индексации запускает особую ситуацию

Range, если ей задан выходящий из диапазона значений индекс.

Конструктор запускает особую ситуацию Size, если ему задан

недопустимый размер вектора:

Vector::Vector(int sz)

{

if (sz<0 || max<sz) throw Size();

// ...

}

Пользователь класса Vector может различить эти две особые ситуации,

если в проверяемом блоке (т.е. в блоке оператора try) укажет

обработчики для обеих ситуаций:

void f()

{

try {

use_vectors();

}

catch (Vector::Range) {

// ...

}

catch (Vector::Size) {

// ...

}

}

В зависимости от особой ситуации будет выполняться соответствующий

обработчик. Если управление дойдет до конца операторов обработчика,

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

обработчиков:

void f()

{

try {

use_vectors();

}

catch (Vector::Range) {

// исправить индекс и

// попробовать опять:

f();

}

catch (Vector::Size) {

cerr << "Ошибка в конструкторе Vector::Size";

exit(99);

}

// сюда мы попадем, если вообще не было особых ситуаций

// или после обработки особой ситуации Range

}

Список обработчиков напоминает переключатель, но здесь в теле

обработчика операторы break не нужны. Синтаксис списка обработчиков

отличен от синтаксиса вариантов case переключателя частично по

этой причине, частично потому, чтобы показать, что каждый

обработчик определяет свою область видимости (см. $$9.8).

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

void f1()

{

try {

f2(v);

}

catch (Vector::Size) {

// ...

}

}

void f2(Vector& v)

{

try {

use_vectors();

}

catch (Vector::Range) {

// ...

}

}

Здесь f2() перехватит особую ситуацию Range, возникающую в

use_vectors(), а особая ситуация Size будет оставлена для f1().

С точки зрения языка особая ситуация считается обработанной сразу

при входе в тело ее обработчика. Поэтому все особые ситуации,

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

в функциях, вызвавших ту функцию, которая содержит проверяемый блок.

Значит в следующем примере не возникнет бесконечного цикла:

try {

// ...

}

catch (input_overflow) {

// ...

throw input_overflow();

}

Здесь input_overflow (переполнение при вводе) - имя глобального класса.

Обработчики особых ситуаций могут быть вложенными:

try {

// ...

}

catch (xxii) {

try {

// сложная реакция

}

catch (xxii) {

// ошибка в процессе сложной реакции

}

}

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

и чаще всего она является свидетельством плохого стиля.

9.3 Имена особых ситуаций

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

запускается ведь не тип, а объект. Если нам нужно передать некоторую

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

поместить в запускаемый объект. Например, допустим нужно знать

значение индекса, выходящее за границы диапазона:

class Vector {

// ...

public:

class Range {

public:

int index;

Range(int i) : index(i) { }

};

// ...

int& operator[](int i)

// ...

};

int Vector::operator[](int i)

{

if (o<=i && i <sz) return p[i];

throw Range(i);

}

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

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

void f(Vector& v)

{

// ...

try {

do_something(v);

}

catch (Vector::Range r ) {

cerr << "недопустимый индекс" << r.index << '\n';

// ...

}

// ...

}

Конструкция в скобках после служебного слова catch является по сути

описанием и она аналогична описанию формального параметра функции.

В ней указывается каким может быть тип параметра (т.е. особой

ситуации) и может задаваться имя для фактической, т.е. запущенной,

особой ситуации. Вспомним, что в шаблонах типов у нас был выбор

для именования особых ситуаций. В каждом созданном по шаблону классе

был свой класс особой ситуации:

template<class T> class Allocator {

// ...

class Exhausted { }

// ...

T* get();

};

void f(Allocator<int>& ai, Allocator<double>& ad)

{

try {

// ...

}

catch (Allocator<int>::Exhausted) {

// ...

}

catch (Allocator<double>::Exhausted) {

// ...

}

}

С другой стороны, особая ситуация может быть общей для всех

созданных по шаблону классов:

class Allocator_Exhausted { };

template<class T> class Allocator {

// ...

T* get();

};

void f(Allocator<int>& ai, Allocator<double>& ad)

{

try {

// ...

}

catch (Allocator_Exhausted) {

// ...

}

}

Какой способ задания особой ситуации предпочтительней, сказать трудно.

Выбор зависит от назначения рассматриваемого шаблона.

9.3.1 Группирование особых ситуаций

Особые ситуации естественным образом разбиваются на семейства.

Действительно, логично представлять семейство Matherr, в которое

входят Overflow (переполнение), Underflow (потеря значимости) и

некоторые другие особые ситуации. Семейство Matherr образуют

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

стандартной библиотеки.

Один из способов задания такого семейства сводится к определению

Matherr как типа, возможные значения которого включают Overflow и

все остальные:

enum { Overflow, Underflow, Zerodivide, /* ... */ };

try {

// ...

}

catch (Matherr m) {

switch (m) {

case Overflow:

// ...

case Underflow:

// ...

// ...

}

// ...

}

Другой способ предполагает использование наследования и виртуальных

функций, чтобы не вводить переключателя по значению поля типа.

Наследование помогает описать семейства особых ситуаций:

class Matherr { };

class Overflow: public Matherr { };

class Underflow: public Matherr { };

class Zerodivide: public Matherr { };

// ...

Часто бывает так, что нужно обработать особую ситуацию Matherr

не зависимо от того, какая именно ситуация из этого семейства

произошла. Наследование позволяет сделать это просто:

try {

// ...

}

catch (Overflow) {

// обработка Overflow или любой производной ситуации

}

catch (Matherr) {

// обработка любой отличной от Overflow ситуации

}

В этом примере Overflow разбирается отдельно, а все другие особые

ситуации из Matherr разбираются как один общий случай. Конечно,

функция, содержащая catch (Matherr), не будет знать какую именно

особую ситуацию она перехватывает. Но какой бы она ни была, при

входе в обработчик передаваемая ее копия будет Matherr. Обычно это

как раз то, что нужно. Если это не так, особую ситуацию можно

перехватить по ссылке (см. $$9.3.2).

Иерархическое упорядочивание особых ситуаций может играть важную

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

такое упорядочивание отсутствует, и нужно обработать все особые

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

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

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

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