Стенли Липпман - Язык программирования C++. Пятое издание
- Название:Язык программирования C++. Пятое издание
- Автор:
- Жанр:
- Издательство:Издательский дом Вильямс
- Год:2014
- Город:Москва
- ISBN:978-5-8459-1839-0
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Стенли Липпман - Язык программирования C++. Пятое издание краткое содержание
Вы держите в руках новое издание популярного и исчерпывающего бестселлера по языку программирования С++, которое было полностью пересмотрено и обновлено под
. Оно поможет вам быстро изучить язык и использовать его весьма эффективными и передовыми способами. В соответствии с самыми передовыми и современными методиками изложения материала авторы демонстрируют использование базового языка и его стандартной библиотеки для разработки эффективного, читабельного и мощного кода.
С самого начала этой книги читатель знакомится со стандартной библиотекой С++, ее самыми популярными функциями и средствами, что позволяет сразу же приступить к написанию полезных программ, еще не овладев всеми нюансами языка. Большинство примеров из книги было пересмотрено так, чтобы использовать новые средства языка и продемонстрировать их наилучшие способы применения. Эта книга — не только проверенное руководство для новичков в С++, она содержит также авторитетное обсуждение базовых концепций и методик языка С++ и является ценным ресурсом для опытных программистов, особенно желающих побыстрей узнать об усовершенствованиях С++11.
Стенли Б. Липпман Жози Лажойе Барбара Э. Му • Узнайте, как использовать новые средства языка С++11 и стандартной библиотеки для быстрого создания надежных программ, а также ознакомьтесь с высокоуровневым программированием
• Учитесь на примерах, в которых показаны передовые стили программирования и методики проектирования
• Изучите принципы и узнайте почему язык С++11 работает именно так
• Воспользуйтесь множеством перекрестных ссылок, способных помочь вам объединить взаимосвязанные концепции и проникнуть в суть
• Ознакомьтесь с современными методиками обучения и извлеките пользу из упражнений, в которых подчеркиваются ключевые моменты, позволяющие избежать проблем
• Освойте лучшие методики программирования и закрепите на практике изученный материал
Исходный код примеров можно загрузить с веб-страницы книги на сайте издательства по адресу: http://www.williamspublishing.com
Язык программирования C++. Пятое издание - читать онлайн бесплатно полную версию (весь текст целиком)
Интервал:
Закладка:
void recoup (int) noexcept; // не будет передавать исключений
void alloc(int); // может передавать исключения
Эти объявления заявляют, что функция recoup()
не будет передавать исключений, а функция alloc()
могла бы. Считается, что к функции recoup()
применена спецификация запрета передачи исключения (nonthrowing specification).
Спецификатор noexcept
должен присутствовать во всех объявлениях и в соответствующем определении функции или ни в одном из них. Спецификатор предшествует замыкающему типу (см. раздел 6.3.3). Спецификатор noexcept
можно определить также в объявлении и определении указателя на функцию. Он неприменим к псевдониму типа или определению типа ( typedef
). В функции-члене спецификатор noexcept
следует за квалификатором const
или квалификатором ссылки, но предшествует квалификаторам final
, override
и = 0
у виртуальной функции.
Важно понимать, что компилятор не проверяет спецификацию noexcept
во время компиляции. Фактически компилятору не разрешено отклонять функцию со спецификатором noexcept
просто потому, что она содержит оператор throw
или вызывает функцию, которая может передавать исключение (однако хорошие компиляторы предупреждают о таких случаях):
// эта функция компилируется, хоть она и нарушает свою спецификацию
// исключения
void f() noexcept // обещание не передавать исключений
{
throw exception(); // нарушает спецификацию исключения
}
В результате вполне вероятно, что функция, обещавшая не передавать исключений, фактически передаст его. Если такая функция передаст исключение, для соблюдения обещания во время выполнения вызывается функция terminate()
. Результат прокрутки стека непредсказуем. Таким образом, спецификатор noexcept
следует использовать в двух случаях: если есть уверенность, что функция не будет передавать исключений, или если совершенно неизвестно, как справиться с ошибкой.
Спецификация запрета передачи исключения фактически обещает вызывающей стороне такой функции, что ей не придется иметь дела с исключениями. Функция либо не передаст исключения, либо вся программа закончит работу; в любом случае вызывающей стороне не нести ответственность за исключения.
Во время компиляции компилятор может вообще не проверять спецификации исключения.
У прежних версий языка С++ была более сложная схема спецификаций исключения, позволяющая определять типы исключений, которые могла бы передавать функция. Функция может определить ключевое слово throw
, сопровождаемое заключенным в скобки списком типов, которые могла бы передать функция. Спецификатор throw
располагается в том же месте, где и спецификатор noexcept
в текущем языке.
Этот подход никогда широко не использовался и не рекомендован в текущем стандарте. Хотя один случай использования более сложной старой схемы распространен довольно широко. Функция, обозначенная как throw()
, обещает не передавать никаких исключений:
void recoup(int) noexcept; // recoup() не передает ничего
void recoup(int) throw(); // эквивалентное объявление
Эти объявления функции recoup()
эквивалентны. Оба указывают, что функция recoup()
не будет передавать исключений.
noexcept
Спецификатор noexcept
получает необязательный аргумент, тип которого должен быть преобразуем в тип bool
: если аргументом будет true
, то функция не будет передавать исключений; если false
— то может:
void recoup(int) noexcept(true); // не будет передавать исключений
void alloc(int) noexcept(false); // может передавать исключения
noexcept
Аргументы спецификатора
noexcept
зачастую создаются с использованием оператора noexcept
. Оператор noexcept
— унарный, возвращающий константное логическое выражение r-значения, означающее способность данного выражения передавать исключения. Подобно оператору sizeof
(см. раздел 4.9), оператор noexcept
не вычисляет свой операнд.
Например, следующее выражение возвращает значение true
:
noexcept(recoup(i)) // true, если вызов функции recoup() не может
// передать исключение, и false в противном случае
поскольку функция recoup()
объявлена со спецификатором noexcept
. В более общем виде выражение noexcept(е)
возвращает значение true
, если у всех вызванных е
функций нет спецификаций передачи и сама е
не содержит операторов throw
. В противном случае выражение noexcept(е)
возвращает значение false
.
Оператор noexcept
можно использовать для формирования спецификатора исключения следующим образом:
void f() noexcept(noexcept(g())); // f() имеет тот же спецификатор
// исключения, что и g()
Если функция g()
обещает не передавать исключений, то f()
также не будет. Если g()
не имеет спецификатора исключения или имеет спецификатор, позволяющий передачу исключений, то функция f()
также может передавать их.
Ключевое слово
noexcept
имеет два значения: это спецификатор исключения, когда оно следует за списком параметров функции, и оператор, который зачастую используется как логический аргумент для спецификатора исключения noexcept
.
Хотя спецификатор noexcept
не является частью типа функции, наличие у функции спецификатора исключения влияет на ее использование.
Указатель на функцию и функция, на которую указывает этот указатель, должны иметь одинаковые спецификации. Таким образом, если объявлен указатель со спецификатором запрета передачи исключения, то использовать этот указатель можно только для указания на функции с подобным спецификатором. Указатель на функцию, способную передавать исключение, определенный явно или неявно, может указывать на любую функцию, даже если она обещает не передавать исключения:
// recoup() и pf1() обещают не передавать исключений
void (*pf1)(int) noexcept = recoup;
// ok: recoup() не будет передавать исключений; и не имеет значения,
// что pf2() может
void (*pf2)(int) = recoup;
pf1 = alloc; // ошибка: alloc() может передать исключение, но pf1()
Интервал:
Закладка: