Скотт Мейерс - Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14
- Название:Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14
- Автор:
- Жанр:
- Издательство:Вильямс
- Год:2016
- Город:Москва
- ISBN:978-5-8459-2000-3
- Рейтинг:
- Избранное:Добавить в избранное
-
Отзывы:
-
Ваша оценка:
Скотт Мейерс - Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14 краткое содержание
В книге рассматриваются следующие темы. Освоение С++11 и С++14 — это больше, чем просто ознакомление с вводимыми этими стандартами возможностями (например, объявлениями типов
, семантикой перемещения, лямбда-выражениями или поддержкой многопоточности). Вопрос в том, как использовать их эффективно, чтобы создаваемые программы были корректны, эффективны и переносимы, а также чтобы их легко можно было сопровождать. Именно этим вопросам и посвящена данная книга, описывающая создание по-настоящему хорошего программного обеспечения с использованием C++11 и С++14 — т.е. с использованием современного С++.
■ Преимущества и недостатки инициализации с помощью фигурных скобок, спецификации
, прямой передачи и функций
интеллектуальных указателей
■ Связь между
,
, rvalue-ссылками и универсальными ссылками
■ Методы написания понятных, корректных,
лямбда-выражений
■ Чем
отличается от
, как они используются и как соотносятся с API параллельных вычислений С++
■ Какие из лучших методов “старого” программирования на С++ (т.е. С++98) должны быть пересмотрены при работе с современным С++
Более чем 20 лет книги
серии
являются критерием уровня книг по программированию на С++. Понятное пояснение сложного технического материала принесло ему всемирную известность. Он всегда самый желанный гость на международных конференциях, а его услуги консультанта широко востребованы во всем мире.
Скотт Мейерс Эффективный и современный С++, После изучения основ С++ я перешел к изучению того, как применять С++ в промышленном программировании, с помощью серии книг Скотта Мейерса Эффективный С++. Эффективный и современный С++ — наиболее важная из книг серии, предлагающая ключевые рекомендации, стили и идиомы, позволяющие эффективно использовать современный С++. Вы еще не купили эту книгу? Сделайте это прямо сейчас. Герб Саттер,
глава Комитета ISO по стандартизации С++, специалист в области архитектуры программного обеспечения на С++ в Microsoft
Эффективный и современный С++. 42 рекомендации по использованию С++11 и С++14 - читать онлайн бесплатно полную версию (весь текст целиком)
Интервал:
Закладка:
Пока указатели std::weak_ptr
указывают на управляющий блок (т.e. слабый счетчик больше нуля), этот управляющий блок должен продолжать существовать. А пока существует управляющий блок, память, его содержащая, должна оставаться выделенной. Таким образом, память, выделенная mаkе
-функцией для std::shared_ptr
не может быть освобождена до тех пор, пока не будут уничтожены последний указатель std::shared_ptr
и последний указатель std::weak_ptr
, ссылающиеся на объект.
Если время между уничтожением последнего std::shared_ptr
и последнего std::weak_ptr
значительно, между уничтожением объекта и занимаемой им памятью может происходить задержка, что особенно важно для типов с большим размером:
class ReallyBigType { … };
auto pBigObj = // Создание большого
std::make_shared(); // объекта с помощью
// std::make_shared
… // Создание указателей std::shared_ptr и std::weak_ptr
// на большой объект и работа с ними
… // Уничтожение последнего указателя std::shared_ptr на
// этот объект; остаются указатели std::weak_ptr
… //Во время этого периода память, ранее занятая большим
// объектом, остается занятой
… // Уничтожение последнего указателя std::weak_ptr на
// объект; освобождение памяти, выделенной для
// управляющего блока и объекта
При непосредственном использовании new
память для объекта ReallyBigType
может быть освобождена, как только уничтожается последний указатель std::shared_ptr
, указывающий на него:
class ReallyBigType { … }; // Как и ранее
// Создание очень большого объекта с помощью new:
std::shared_ptr pBigObj( newReallyBigType);
… // Как и ранее, создание указателей std::shared_ptr и
// std::weak_ptr на объект и работа с ними
… // Уничтожение последнего указателя std::shared_ptr на
// этот объект; остаются указатели std::weak_ptr
// Память, выделенная для объекта, освобождается
… //Во время этого периода остается занятой только память,
// ранее выделенная для управляющего блока
… // Уничтожение последнего указателя std::weak_ptr на
// объект; освобождение памяти, выделенной для
// управляющего блока
Оказавшись в ситуации, когда использование функции std::make_shared
невозможно или неприемлемо, вы можете захотеть защититься от ранее рассматривавшихся нами проблем, связанных с безопасностью исключений. Лучший способ сделать это — обеспечить немедленную передачу результата операции new
конструктору интеллектуального указателя в инструкции, которая не делает ничего иного . Это предотвратит создание компилятором кода, который может генерировать исключение между оператором new
и вызовом конструктора интеллектуального указателя, который будет управлять объектом, созданным оператором new
.
В качестве примера рассмотрим небольшое изменение небезопасного с точки зрения исключений вызова функции processWidget
, которую мы рассматривали ранее. В этот раз мы укажем пользовательский удалитель:
void processWidget(std::shared_ptr spw, // Как и ранее
int priority);
void cusDel(Widget *ptr); // Пользовательский удалитель
Вот небезопасный с точки зрения исключений вызов:
processWidget( // Как и ранее,
std::shared_ptr(new Widget,cusDel), // потенциальная
computePriority() // утечка
); // ресурса!
Вспомним: если computePriority
вызывается после new Widget
, но до конструктора std::shared_ptr
и если computePriority
генерирует исключение, происходит утечка динамически созданного Widget
.
Здесь применение пользовательского удалителя препятствует использованию std::make_shared
, так что избежать проблемы можно, поместив создание Widget
в динамической памяти и конструирование std::shared_ptr
в собственную инструкцию, а затем вызвав функцию processWidget
с передачей ей полученного std::shared_ptr
. Вот и вся суть этого метода, хотя, как мы вскоре увидим, его можно подкорректировать для повышения производительности:
std::shared_ptr spw(new Widget, cusDel);
processWidget(spw, computePriority()); // Корректно, но не
// оптимально; см. ниже
Этот код работает, поскольку std::shared_ptr
предполагает владение обычным указателем, переданным конструктору, даже если этот конструктор генерирует исключение. В данном примере, если конструктор spw
генерирует исключение (например, из-за невозможности выделить динамическую память для управляющего блока), он все равно гарантирует вызов cusDel
для указателя, полученного в результате операции new Widget
.
Небольшое снижение производительности возникает из-за того, что в небезопасном с точки зрения исключений коде мы передавали функции processWidget
rvalue.
processWidget (
std::shared_ptr(new Widget, cusDel), // rvalue
computePriority()
);
а в случае безопасного вызова — передаем lvalue:
processWidget( spw, computePriority()); // lvalue
Поскольку параметр std::shared_ptr
функции processWidget
передается по значению, создание из rvalue
влечет за собой только перемещение, в то время как создание из lvalue требует копирования. Для указателя std::shared_ptr
разница может оказаться существенной, так как копирование std::shared_ptr
требует атомарного инкремента счетчика ссылок, в то время как перемещение std::shared_ptr
не требует никаких действий со счетчиком ссылок вообще. Чтобы безопасный с точки зрения исключений код достиг уровня производительности небезопасного кода, нам надо применить std: :move
к spw
для того, чтобы превратить его в rvalue
(см. раздел 5.1):
processWidget( std::move(spw ), // Эффективно и безопасно
computePriority()); // в смысле исключений
Это интересный метод, и его стоит знать, но обычно это не имеет особого значения, поскольку вы будете редко сталкиваться с причинами не использовать make
-функцию. Если у вас нет убедительных причин поступать иначе, используйте make
-функции.
• По сравнению с непосредственным использованием new
, make
-функции устраняют дублирование кода, повышают безопасность кода по отношению к исключениям и в случае функций std::make_shared
и std::allocate_shared
генерируют меньший по размеру и более быстрый код.
• Ситуации, когда применение makе
-функций неприемлемо, включают необходимость указания пользовательских удалителей и необходимость передачи инициализаторов в фигурных скобках.
• Для указателей std::shared_ptr
дополнительными ситуациями, в которых применение makе-функций может быть неблагоразумным, являются классы с пользовательским управлением памятью и системы, в которых проблемы с объемом памяти накладываются на использование очень больших объектов и наличие указателей std::weak_ptr
, время жизни которых существенно превышает время жизни указателей std::shared_ptr
.
Интервал:
Закладка: